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编写 背景 
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“软件 项 目 开发 全 程 实录 ”丛书 〈 第 1 版 ) 出 版 两 年 来 ， 受 到 了 广大 读者 的 热烈 欢迎 ， 目 前 该 丛 
书 已 经 累计 销售 近 10 万 册 ， 成 为 近年 来 最 畅销 的 编程 案例 类 丛书 之 一 。 应 读者 朋友 的 要 求 ， 我 们 根据 
最 新 的 市 场 变化 并 结合 广大 读者 的 建议 ， 编 写 了 该 丛书 的 第 2 版 。 

从 技术 背景 来 看 ，PHP 是 全 球 最 普及 、 应 用 最 广泛 的 互联 网 开发 语言 之 一 ， 它 有 开放 的 源 代 码 ， 
独特 的 语法 结构 ， 包 含 了 C、Java、Perl 等 语言 的 特点 ， 具 有 多 种 数据 库 的 支持 ， 并 且 支持 跨 平 台 的 操 
作 ， 支 持 面向 对 象 的 编程 ， 而 且 是 完全 免费 的 。PHP 越 来 越 受到 广大 程序 员 的 青睐 和 认同 ， 正 在 逐渐 
成 为 程序 开发 者 使 用 的 主流 语言 。 

为 了 帮助 众多 学 习 PHP 的 读者 提高 PHP 项 目 开发 水 平 , 笔者 精心 编著 了 本 书 。 本 书 充分 利用 PHP 
在 应 用 程序 的 设计 和 开发 上 的 灵活 高 效 的 特点 ， 以 12 个 实用 的 网 络 程序 为 案例 ， 详 细 地 讲解 整个 项 目 
的 开发 过 程 。 


本 书 内 容 


本 书 共 12 章 。 第 1、2、3、4、6、7、8 章 结合 目前 最 常用 的 MySQL 5.0 数据 库 分 别 开 发 了 九 九 
度 供 求 信息 网 、BCTY365 网 上 社区 系统 、 办 公 自 动 化 管理 系统 、 电 子 商务 平台 、 物 流 配 送信 息 网 、 学 
校 图 书馆 管理 系统 和 博客 管理 系统 ; 第 5 章 应 用 目前 最 流行 的 Ajax 技术 结合 最 常用 的 SQL Server 2000 
数据 库 开发 在 线 考试 网 ， 从 而 实现 一 个 无 刷新 的 在 线 考试 系统 ; 第 9 章 结合 最 新 的 Access 2003 数据 库 
开发 企业 快 信 网 ， 其 中 应 用 Web Service 服务 实现 短信 的 发 送 ， 应 用 imap 电子 邮件 系统 函数 实现 邮件 
的 收发 ， 第 10 章 结 合 目前 流行 的 ADODB 类 库 开发 支持 不 同 数据 库 的 Online 影视 365 网 ， 可 以 根据 
用 户 的 需求 更 换 不 同类 型 的 数据 库 ; 第 11 章 应 用 Zend Framework 框架 开发 一 个 明日 知道 网 站 ， 打 造 
一 个 更 加 专业 化 的 编程 技术 搜索 引擎 。 第 12 章 应 用 Smarty 模板 和 ADODB 类 库 技术 开发 一 个 电子 商 
务 网 站 。 


第 2 版 所 做 的 改进 


在 第 2 版 中 ， 我 们 主要 遵循 以 下 原则 对 第 1 版 内 容 进 行 修改 。 
从 增加 了 内 容 
针对 目前 PHP 语言 得 到 越 来 越 广泛 的 应 用 的 特点 ， 我 们 增加 明日 知道 搜索 引擎 和 电子 商务 网 站 两 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


个 系统 ， 以 便 读 者 更 好 地 学 习 PHP 知识 。 

他 使 用 最 新 技术 

在 第 1 版 中 项 目 开发 应 用 的 是 PHP、MySQL、SQL Server 和 ADODB 等 技术 ， 在 第 2 版 中 引入 
Zend Framework 框架 、Smarty 模板 和 Ajax 技术 ， 开 发 目前 最 具 实用 性 的 Web 程序 ， 以 适应 读者 的 
需要 。 

总 修改 了 第 1 版 中 的 错误 

对 于 第 1 版 中 出 现 的 错误 ， 或 者 运行 异常 的 程序 进行 修订 ， 以 便 读 者 更 好 地 学 习 。 


本 书 特点 


仿 视频 讲解 

对 于 初学 者 来 说 ， 视 频 讲 解 是 最 好 的 导师 ， 它 能 够 引导 初学 者 快速 入 门 ， 使 初学 者 感受 到 编程 的 
快乐 和 成 就 感 ， 进 一 步 增强 学 习 的 信心 。 鉴 于 此 ， 本 书 为 每 一 个 案例 都 配备 了 视频 讲解 ， 初 学 者 可 以 
通过 视频 讲解 实现 案例 中 的 功能 。 

续 案例 典型 

本 书 案例 均 从 实际 应 用 角度 出 发 ， 应 用 了 当前 流行 的 技术 ， 涉 及 的 知识 广泛 ， 读 者 可 以 从 每 个 案 
例 中 积累 丰富 的 实战 经 验 。 

人 代码 注释 

为 了 便于 读者 阅读 程序 代码 ， 书 中 的 代码 几乎 每 行 都 有 注释 ， 并 且 整 齐 地 纵向 排列 ， 使 读者 快速 
地 领会 作者 的 意图 。 

< 代码 贴 十 

案例 类 书籍 通常 会 包含 大 量 的 程序 代码 ， 元 长 的 代码 往往 令 初 学 者 望 而 生 黑 。 为 了 方便 读者 阅读 
和 理解 代码 ， 本 书 避 免 了 连续 大 篇 幅 的 代码 ， 将 其 分 割 为 多 个 部 分 ， 并 对 重要 的 变量 、 方 法 和 知识 点 
设计 了 颇具 特色 的 “代码 贴 士 ”。 

总 知识 扩展 

为 了 增加 读者 的 编程 经 验 和 技巧 ， 书 中 每 个 案例 都 标记 有 “注意 ”、“ 技 巧 ”等 提示 信息 ， 并 且 
在 每 章 内 容 中 都 提供 一 项 专题 技术 。 


适合 读者 


本 书 定位 于 想 要 使 用 PHP 进行 网 站 开发 的 初中 级 读者 ， 一 些 案例 对 于 高 级 用 户 也 具有 很 好 的 参考 
价值 。 同 时 ， 本 书 也 特别 适合 作为 大 专 院 校 计算 机 专业 广大 师 生 的 案例 参考 书 。 对 于 初学 者 来 说 不 必 
担心 书 中 的 案例 做 不 出 来 ， 因 为 本 书 附带 光盘 中 的 视频 录像 会 引导 您 去 做 ， 再 结合 书 中 的 知识 讲解 ， 
相信 初学 者 不 但 会 做 案例 程序 ， 而 且 能 够 根据 理解 自行 设计 类 似 的 程序 。 


本 书 约定 


由 于 篇 幅 有 限 ， 本 书 每 章 并 不 能 逐一 介绍 案例 中 的 各 模块 。 作 者 选择 其 中 基础 和 典型 的 模块 进行 
介绍 ， 对 于 功能 重复 的 模块 ， 由 于 技术 、 设 计 思路 和 实现 过 程 基本 相同 ， 因 此 没有 在 书 中 体现 。 
本 书 中 涉及 的 功能 模块 在 光盘 中 都 附带 有 视频 录像 ， 方 便 读 者 学 习 。 


作者 队伍 


本 书 由 明日 科技 PHP 软件 开发 团队 组 织 编写 。 明日 科技 是 一 家 以 计算 机 软件 技术 为 核心 的 高 科技 
企业 ， 是 专业 的 应 用 软件 开发 和 服务 企业 ， 多 年 来 始终 致力 于 行业 管理 软件 开发 、 行 业 电 子 商务 网 站 
开发 等 领域 ， 涉 及 生产 、 管 理 、 控 制 、 仓 贮 、 物 流 、 营 销 、 服 务 等 行业 。 另 外 ， 公 司 还 致力 于 软件 技 
术 的 普及 和 传播 。 

参与 本 书 编写 的 程序 员 有 : 刘 中 华 、 潘 凯 华 、 杨 明 、 董 大 水 、 尹 强 、 李 继 业 、 张 舌 、 赛 硅 春 、 高 
春 攀 、 宋 坤 、 刘 锐 宁 、 梁 水 、 李 伟 明 、 王 小 科 、 王 军 、 赵 会 东 、 刘 彬 彬 、 安 剑 、 孙 秀 梅 、 巩 建华 、 房 
大 伟 、 吕 双 、 刘 云 峰 、 王 国 辉 、 李 钟 尉 、 张 振 坤 、 陈 丹 丹 、 卢 翰 、 白 伟 明 、 刘 欣 、 李 慧 、 梁 晓 岗 、 杨 
丽 、 顾 丽 丽 、 刘 龄 龄 、 于 长 风 、 张 彦 国 、 王 乐 乐 、 陈 英 、 刘 莹 、 李 贺 等 。 


本 书 服务 


为 了 更 好 地 为 读者 朋友 服务 ， 我 们 提供 了 以 下 服务 联系 方式 : 

服务 网 站 : www.mingribook.com 

服务 信箱 : mingrisoft@mingrisoft.com、tmoonbook@sina.com、th_press@263.net 

客服 电话 : 0431-84978981 84978982 转 802、803 

在 编写 本 书 的 过 程 中 ， 我 们 本 着 科学 、 严 说 的 态度 ， 力 求 精 益 求 精 ， 但 错误 、 朴 漏 之 处 在 所 难免 ， 
敬 请 广大 读者 批评 指正 。 读 者 在 阅读 本 书 时 ， 如 果 发 现 错误 或 遇 到 问题 ， 可 以 通过 发 送 电 子 邮 件 、 论 
坛 发 帖 及 拨打 客服 电话 等 方式 与 我 们 联系 ， 我 们 承诺 在 1 一 5 个 工作 日 内 给 您 回复 。 

最 后 ， 感 谢 您 购买 本 书 ， 希 望 本 书 能 成 为 您 的 良师益友 ， 成 为 您 步 入 编程 高 手 之 路 的 踏 脚 石 。 

宝剑 锋 从 磨 硕 出 ， 梅 花香 自 苦 寒 来 。 

祝 读书 快乐 ! 
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第 章 


度 供求 信息 网 


( OE 5.0 实现 ) 
(器 " 视频 讲解 : 1 小 时 50 分钟) 


在 全 球 知 识 经 济 和 信息 化 高 速 发 展 的 今天 ， 信 息 化 是 决定 企业 成 败 的 关键 因 
素 , 企业 需要 在 网 站 上 发 布 供求 信息 , 以 促使 企业 在 同 领域 中 得 到 突飞猛进 的 发 展 。 

一 个 广泛 的 、 快 速 的 、 自 由 的 信息 交流 平台 ， 为 用 户 带 来 方便 的 同时 ， 也 会 给 
企业 带 来 无 限 的 商机 。 于 是 ， 以 因 将 网 为 基础 的 信息 交流 平台 即 九 九 度 供求 信息 网 
出 现 了 。 九 九 度 供求 信息 网 致力 于 优化 信息 交流 ， 实 现 信息 的 快速 交流 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


使 当前 窗口 承载 框架 页 中 的 起 链接 页 面 

如 何 自动 计算 愉 系 统 日 期 为 基数 的 相对 日 期 
do…While 循环 语句 的 应 用 

查询 关键 字 描 红 技术 

在 Windows 操作 系统 下 搭建 PHP 开发 环境 
在 Windows 操作 系统 下 创建 数据 库 和 数据 表 
PowerDesigner 建 模 的 应 用 

单元 测试 技术 

框架 技术 在 Web 网 站 中 的 应 用 

如 何 发 布 网 站 

表单 数据 的 两 种 提交 方式 

应 用 phpMyAdmin 工具 创建 和 删除 数据 库 、 数 据 胡 


浊音 音 音 音 间 总 总 总 总 吾 吾 
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Wl 开交 少 水 


XX x 信息 科技 有 限 公 司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 为 了 扩大 企业 规模 ， 增 强 企 
业 的 竞争 力 ， 该 公司 决定 向 多 元 化 方向 发 展 ,借助 Internet 在 国内 的 快速 发 展 ， 聚 集 部 分 资金 投入 网 站 
建设 ， 为 企业 和 用 户 提供 综合 信息 服务 ， 以 向 企业 提供 有 偿 信 息 服务 为 一 利 方式 ， 打 造 一 个 全 新 的 供 
求 信息 网 。 例 如 ， 提 供 企 业 广告 、 发 布 各 类 免费 供求 信息 、 发 布 企 业 付费 信息 等 服务 方式 。 现 需要 委 
托 其 他 单位 开发 一 个 综合 信息 网 站 。 


1.2 系统 分 析 


1.2.1 需求 分 析 


对 于 信息 网 站 来 说 ， 用 户 的 访问 量 是 至 关 重要 的 。 如 果 网 站 的 访问 量 很 低 ， 那 么 就 很 少 有 企业 会 
要 求 为 他 提供 有 偿 服 务 ， 也 就 没有 利润 可 言 了 。 因 此 信息 网 站 必须 为 用 户 提供 大 量 的 、 免 费 的、 有 价 
值 的 信息 才能 够 吸引 用 户 。 为 此 ， 网 站 不 仅 要 为 企业 提供 各 种 有 偿 服务 ， 还 需要 额外 为 用 户 提供 大 量 
的 无 偿 服 务 。 通 过 与 企业 的 实际 接触 和 沟通 ， 确 定 网 站 应 包括 招聘 信息 、 求 职 信息 、 培 训 信息 、 公 寓 
信息 、 家 教 信息 、 车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 竞 出 竞 ， 寻 求 合作 、 企 业 广告 等 服务 。 

通过 实际 调查 ， 要 求 供求 信 息 网 具有 以 下 功能 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 
实现 强大 的 供求 信息 查询 ， 支 持 模糊 查询 。 
用 户 不 需要 注册 ， 便 可 免费 发 布 供求 信息 。 
免费 发 布 的 供求 信息 必须 经 后 台 审 核 后 才能 正式 发 布 ， 避 免 不 良 信息 。 
支持 海量 数据 录入 。 
由 于 供求 信息 数据 量 大 ， 后 台 应 该 可 以 随时 清理 数据 。 


办 办 办 办 办 罗 


1.2.2 ”可行 性 分 析 


根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 可 行 性 分 析 的 要 求 ， 制 定 可 行 性 研究 报 
告 如 下 : 

1. 引言 

(1) 编写 目的 

为 了 给 企业 的 决策 层 提供 是 否 进行 项 目 实 施 的 参考 依据 ， 现 以 文件 的 形式 分 析 项 目的 风险 、 项 目 
需要 的 投资 与 效益 。 

(2) 背景 
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X x Xx 信息 科技 有 限 公司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 企 业 为 了 不 断 满足 客户 的 需 
求 ， 为 达到 企业 在 同行 业 领 域 中 的 领先 地 位 ， 现 需要 委托 其 他 公司 开发 一 个 综合 信息 网 ， 项 目 名 称 为 
九 九 度 供求 信息 网 。 

2. 可 行 性 研究 的 前 提 

(1) 要 求 

九 九 度 供求 信息 网 要 求 能 够 提供 信息 搜索 、 信 息 定位 描 红 、 发 布 免费 信息 、 发 布 付费 信息 、 发 布 
企业 广告 和 对 各 类 发 布 的 信息 进行 审核 、 删 除 、 检 索 等 功能 。 

(2) 目标 

九 九 度 供求 信息 网 的 主要 目标 是 提供 强大 的 搜索 功能 ， 准 确 的 信息 描 红 定位 功能 ， 付 费 信息 的 管 
理 ， 免 费 信息 的 审核 和 删除 功能 。 

(3) 条 件 、 假 定 和 限制 

项 目 需 要 在 两 个 月 内 交付 用 户 使 用 。 系 统 分 析 师 需要 3 天 内 到 位 ， 用 户 需 要 4 天 时 间 确 认 需 求 分 
析 文 档 。 去 除 员工 两 个 月 的 正常 休息 日 16 天 , 那么 程序 开发 人 员 需 要 在 1 个 月 零 几 天 的 时 间 内 进行 系 
统 设 计 、 程 序 编码 、 系 统 测 试 、 程 序 调试 和 网 站 部 署 工作 。 

(4) 评价 尺度 

根据 用 户 的 要 求 ， 系 统 应 以 搜索 引擎 为 主 ， 对 于 发 布 的 供求 信息 应 能 及 时 准确 地 保存 、 审 核 、 查 
询 、 描 红 定 位 。 由 于 用 户 存在 多 个 营业 点 ， 系 统 应 具有 局 域 网 操作 的 能 力 ， 在 多 个 营业 点 同时 运行 系 
统 时 ， 系 统 中 各 项 操作 的 延 时 不 能 超过 10 秒 钟 。 此 外 ， 在 系统 出 现 故 障 时 ， 应 能 及 时 进行 恢复 。 

3， 投资 及 效益 分 析 

(1) 支出 

根据 系统 的 规模 及 两 个 月 的 项 目 开发 周期 ， 公 司 决定 投入 5 个 人 。 因 此 ， 公 司 将 直接 支付 8 万 元 
的 工资 及 各 种 福利 待遇 。 在 项 目 安装 及 调试 阶段 ， 用 户 培训 、 员 工 出 差 等 费用 支出 需要 2 万 元 。 在 项 
目 维护 阶段 预计 需要 投入 2 万 元 的 资金 ， 累 计 项 目 投入 需要 12 万 元 资金 。 

(2) 收益 

用 户 提供 项 目 资金 30 万 元 。 对 于 项 目 运行 后 进行 的 改动 , 采取 协商 的 原则 根据 改动 规模 额外 提供 
资金 。 因 此 从 投资 与 收益 的 效益 比 上 ， 公 司 可 以 获得 18 万 元 的 利润 。 

项 目 完成 后 ， 将 给 公司 提供 资源 储备 ， 包 括 技术 、 经 验 的 积累 ， 以 后 再 开发 类 似 的 项 目 时 ， 可 以 
极 大 地 缩短 项 目 开 发 周期 。 

4. 结论 

根据 上 面 的 分 析 ， 技 术 上 不 会 存在 问题 ， 因 此 项 目 延 期 的 可 能 性 很 小 。 在 效益 上 ， 公 司 投入 5 个 
人 、 两 个 月 的 时 间 获 利 18 万 元 ， 比 较 可 观 。 在 公司 今后 的 发 展 上 可 以 储备 网 站 开发 的 经 验 和 资源 ， 因 
此 认为 该 项 目 可 以 开发 。 


1.2.3 ”编写 项 目 计划 书 


根据 《4GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 的 项 目 开发 计划 要 求 ， 结 合 单位 实际 情 
况 ， 设 计 项 目 计划 书 如 下 : 
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1. 引言 

(1) 编写 目的 

为 了 保证 项 目 开 发 人 员 按 时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合理 的 顺序 开 
展 工作 ， 现 以 书面 的 形式 将 项 目 开发 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工作 责 
任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描述 出 来 ， 作 为 项 目 相 关 人 员 之 间 的 共 
识 和 约定 以 及 项 目 周期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 

九 九 度 供求 信息 网 是 本 公司 与 X X X 信 息 科技 有 限 公司 签订 的 待 开发 项 目 ， 网 站 性 质 为 信息 服务 
类 型 ， 可 为 信息 发 布 者 有 偿 或 无 偿 地 提供 招聘 、 求 职 、 培 训 、 求 购 、 公 寓 、 车 辆 、 房 屋 和 出 售 等 信息 。 
项 目 周期 为 两 个 月 ， 项 目 背景 规划 如 表 1.1 所 示 。 


表 1.1 项 目 背景 规划 
项 目 名 称 签订 项 目 单位 项 目 负责 人 项 目 承 担 部 门 


九 九 度 供求 信息 网 开发 部 站 
性。xx x 网络 和 有 限 全 到 让 
2， 概 述 
(1) 项 目 目标 


项 目 目标 应 当 符合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。 九 九 度 供求 信息 
网 的 项 目 目标 如 下 : 

九 九 度 供求 信息 网 主要 用 来 为 用 户 提供 信息 服务 ， 对 于 生活 和 工作 中 的 各 类 信息 都 应 尽 可 能 地 全 
部 包括 在 内 ， 例 如 公寓 、 求 职 、 招 聘 、 培 训 、 招 商 、 房 屋 、 车 辆 、 出 售 、 求 购 等 信息 。 项 目 发 布 后 
要 实现 能 够 为 用 户 生活 、 工 作 带 来 极 大 的 方便 并 提高 企业 知名 度 、 为 企业 产品 宣传 节约 大 量 成 本 的 目 
标 。 整 个 项 目 需要 在 两 个 月 的 期 限 结束 后 ， 交 给 客户 进行 验收 。 

(2) 产品 目标 与 范围 

一 方面 九 九 度 供求 信息 网 能 够 为 企业 节省 大 量 人 力 资源 , 企业 不 再 需要 大 量 的 业务 人 员 去 跑 市 场 ， 
间接 为 企业 节约 了 成 本 。 另 一 方面 ， 九 九 度 供求 信息 网 能 够 收集 海量 供求 信息 ， 将 会 有 大 量 用 户 访问 
网 站 ， 有 助 于 提高 企业 知名 度 。 

(3) 应 交付 成 果 

项 目 开发 完成 后 ， 交 付 的 内 容 如 下 : 

回 ”以 光盘 的 形式 交付 九 九 度 供求 信息 网 的 源 程序 、 网 站 数据 库 文件 、 系 统 使 用 说 明 书 。 

回 ”客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 、 安 装 PHP 开发 环境 、 协 助 甲 

方 购买 域名 ， 将 开发 的 九 九 度 供求 信息 网 发 布 到 互联 网 上 运行 。 
回 ”网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 
维护 与 服务 。 

(4) 项 目 验收 方式 与 依据 

项 目 验收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
试 员 根据 用 户 需 求 和 项 目 目标 进行 验收 。 项 目 在 通过 内 部 验收 后 交 给 用 户 进行 验收 ， 验 收 的 主要 依据 


可 
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为 需求 规格 说 明 书 。 

3. 项 目 团 队 组 织 

(1) 组 织 结构 

为 了 完成 九 九 度 供求 信息 网 的 项 目 开发 ， 公 司 组 建 了 一 个 临时 的 项 目 团队 ， 由 项 目 经 理 、 系 统 分 
析 师 、PHP 开发 工程 师 、 网 页 设计 师 和 系统 测试 员 构 成 ， 如 图 1.1 所 示 。 


入 


项 目 经 理 


系统 分 析 师 。 PHP 开发 工程 师 网 页 设计 师 系统 测试 员 
图 1.1 项 目 团队 组 织 结构 图 


(2) 人员 分 工 
为 了 明确 项 目 团 队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分 工 表 ， 如 表 1.2 所 示 。 
表 1.2 人 员 分 工 表 
姓 名 | 技术 水 平 工作 描述 
负责 项 目的 审批 、 决 策 的 实施 、 项 目的 前 其 
张 达 明 | MBA 项 目 开发 部 | 项 目 经 理 分 析 和 策划 、 项 目 开发 进度 的 跟踪 、 项 目 质 
量 的 检查 


周 兴 伟 负责 系统 功能 分 析 、 系 统 框架 设计 

分 紫 下 负责 软件 前 后 台 设 计 与 编码 

王 惠子 负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 
舒心 怡 对 软件 进行 测试 、 编 写 软件 测试 文档 


1.3 系统 设计 


1.3.1 系统 目标 


根据 需求 分 析 的 描述 以 及 与 用 户 的 沟通 ， 现 制定 网 站 实现 目标 如 下 : 

回 ”系统 采用 人 机 对 话 方式 ， 界 面 美观 友好 ， 界 面 简洁 、 框 架 清晰 、 美 观 大 方 。 
回 ”灵活 快速 地 填写 供求 信息 ， 使 信息 传递 更 快捷 。 

回 ”信息 查询 灵活 、 方 便 ， 数 据 存储 安全 可 靠 。 
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实施 强大 的 后 台 审核 功能 。 

实现 强大 的 搜索 引擎 ， 支 持 模糊 查询 、 关 键 字 描 红 功 能 等 。 

对 用 户 输入 的 数据 ， 系 统 进行 严格 的 数据 检验 ， 尽 可 能 排除 人 为 的 错误 。 

网 站 最 大 限度 地 实现 易 维护 性 和 易 操作 性 。 

为 充分 展现 网 站 的 交互 性 ， 供 求 信息 网 采用 动态 网 页 技术 实现 用 户 信息 在 线 发 布 。 
具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 


办 办 办 多国 多 
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DD 


系统 功能 结构 


九 九 度 供求 信息 网 前 台 功 能 结构 图 如 图 1.2 所 示 


图 1.2 九 九 度 供求 信息 网 前 台 功 能 结构 图 
九 九 度 供求 信息 网 后 台 功 能 结构 图 如 图 1.3 所 示 。 


发 布 企 
\ 7D 
和 布 付 痪 
1 信息 
人 
并 信和 a 2 
(Care) 所 一 一 | 供求 信息 网 后 台 | 一 一 > 


Fm | 


图 1.3 九 九 度 供求 信息 网 后 台 功 能 结构 图 
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1.3.3 ”系统 流程 图 


九 九 度 供求 信息 网 的 系统 流程 如 图 1.4 所 示 。 


发 布 企业 广告 信息 


图 1.4 系统 流程 图 
1.3.4 ”系统 预览 


九 九 度 供求 信息 网 由 多 个 程序 页 面 组 成 。 下 面 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

前 台 首 页 如 图 1.5 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 查询 、 企 业 广告 信息 显示 、 后 台 登 录入 口 等 功 
能 。 搜 索引 擎 页 面 如 图 1.6 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 快速 检索 、 查 询 关键 字 描 红 等 功能 。 

发 布 免费 信息 页 面 如 图 1.7 所 示 ， 该 页 面 用 于 实现 发 布 分 类 的 免费 信息 功能 。 付 费 管理 页 面 如 
图 1.8 所 示 ， 该 页 面 用 于 实现 付费 信息 分 类 查看 、 付 费 信息 审核 、 付 费 信 息 删 除 等 功能 。 

免费 管理 页 面 如 图 1.9 所 示 ， 该 页 面 用 于 实现 免费 的 信息 分 类 查看 、 信 息 审核 和 信息 删除 等 功能 。 
管理 员 登 录 页 面 如 图 1.10 所 示 ， 该 页 面 用 于 实现 对 管理 员 登 录 的 用 户 名 和 密码 进行 验证 等 功能 。 
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CC 


图 1.5 前 台 首 页 (光盘 \TMW1\99purseyindex.php) ”图 1.6 搜索 引擎 (光盘 \TM\01\99pursey\admin\findinfo.php) 
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图 1.9 免费 管理 (光盘 \TM\01\99pursey\admin\finddd.php) 图 1.10 管理 员 登 录 ( 光 盘 \TM\01\99pursey\admin\login.php) 


1.3.5 ”开发 环境 


在 开发 九 九 度 供求 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 


% 
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= 


.服务 器 端 
操作 系统 : Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.2。 
PHP 软件 : PHP 5.1.6。 
数据 库 : MySQL 5.0.24。 
MySQL 图 形 化 管理 软件 : phpMyAdmin-2.9.0.2。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 
客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 1024X768 像素 。 


回回 轿 轿 网 罗网 加 


回回 


1.3.6 文件 夹 组 织 结构 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 (例如 ， 创 建 一 个 名 为 admin 的 文 
件 夹 ， 用 于 保存 网 站 的 后 台 文 件 ) ， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 架构 。 
笔者 在 开发 九 九 度 供求 信息 网 时 ， 设 计 了 如 图 1.11 所 示 的 文件 夹 架构 图 。 在 开发 时 ， 只 需要 将 所 创建 
的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


用 于 存储 网 站 后 台 文件 

用 于 存储 网 站 后 台 使 用 的 图 片 资源 
用 于 存储 数据 库 连 接 文件 
i CSS 样式 表 


月 于 本 双人 向 
用 于 存储 网 站 使 用 的 自 定义 函数 


图 1.11 文件 夹 组 织 结构 
1.4 在 Windows 操作 系统 下 搭建 PHP 开发 环境 


PHP 能 否 高 效 、 稳 定 地 运行 依赖 于 服务 器 的 编译 和 执行 ， 本 节 主 要 介绍 如 何在 微软 的 Windows 操 
作 系 统 中 架设 安全 、 可 靠 的 PHP 运行 环境 。 


1.4.1 在 Windows 下 应 用 AppServ 快速 配置 PHP 开发 环境 


AppServ 是 PHP 网 页 架 站 工具 组 合 包 , 可 以 将 网 络 上 免费 的 架 站 资源 重新 包装 成 单一 的 安装 程序 。 
它 提供 了 简易 、 快 速 的 PHP 运行 环境 的 搭建 机 制 ， 读 者 只 需 按 照 普 通 应 用 软件 的 安装 方式 就 可 以 完成 
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Apache+MySQL+PHP+phpMyAdmin 的 安装 与 配置 工作 。 

下 面 以 AppServ-win32-2.5.7 为 例 来 介绍 AppServ 的 安装 和 使 用 方法 。 

安装 AppServ 之 前 应 从 官方 网 站 http://www.appservnetwork.com 下 载 最 新 版 本 的 AppServ-win32- 
2.5.7.exe 安装 程序 。 


鳃 中 视频 录像 : TM\O1\Ix\ 在 Windows 下 应 用 AppServ 快速 配置 PHP 开发 环境 .swf 


在 Windows 下 应 用 AppServ 快速 配置 PHP 开发 环境 的 操作 步骤 如 下 : 
(1) 双击 AppServ-win32-2.5.7.exe 文件 ， 打 开 如 图 1.12 所 示 的 AppServ 启动 窗口 。 
(2) 单 击 图 1.12 中 的 Next 按钮 ， 打 开 如 图 1.13 所 示 的 AppServ 安装 协议 窗口 。 


Welcome to the AppServ 25.7 
Setup Wizard 


aymlhaohugjatiWGY 


Everyone verbatim copies 
of this bcense document, but chengng k 5 not slowed, 


ls the first released version of the Lesser GPL Ht also counts 


1.12 ”AppServ 启动 窗口 1.13 ”AppServ 安装 协议 窗口 


(3) 单 击 图 1.13 中 的 IAgree 按钮 ,打开 如 图 1.14 所 示 的 窗口 ， 在 该 窗口 中 可 以 设置 AppServ 的 
安装 路 径 〈 默 认 安装 路 径 一 般 为 C:\AppServ) 。AppServ 安装 完成 后 ，Apache、MySQL、PHP 都 将 以 
子 目录 的 形式 存储 到 该 目录 下 。 

(4) 单 击 图 1.14 中 的 Next 按钮 ， 打 开 如 图 1.15 所 示 的 窗口 ， 在 该 窗口 中 可 以 选择 要 安装 的 程序 
和 组 件 〈 默 认为 全 选 状 态 ) 。 


Select Components 
Select the components you want to instal, ceer the components 
You do not want to instal 


| EF Choose Install Location 
CChoose the older In which to netal AppServ 2.5.7. 


设置 AppServ 的 安装 路 径 


1.14 ”AppServ 安装 路 径 选 择 1.15 AppServ 安装 选项 


®_ 
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(5) 在 图 1.15 中 单 击 Next 按钮 ， 打 开 如 图 1.16 所 示 的 窗口 ， 该 窗口 主要 设置 Apache 的 端口 号 。 
(6) 单 击 图 1.16 中 的 Next 按钮 ， 打 开 如 图 1.17 所 示 的 窗口 。 该 窗口 主要 对 MySQL 数据 库 的 


root 用 户 的 登录 密码 及 字符 集 进行 设置 , 这 里 将 字符 集 设置 为 GB2312 Simplified Chinese, 表示 MySQL 
数据 库 的 字符 集 将 采用 简体 中 文 形式 。 


Apache HTTP Server Infommation 
Please enter your server's nformation ， 


WS ol Password Support (PHP MySQL APIfuncaon ) 
厂 Enable InnoDe 
Nuleoft frstal System v2.19 一 


Nast tet Stem v.19 


ce [aa om | 


图 1.16 Apache 端口 号 设置 


(7) 单 击 图 1.17 中 的 Install 按钮 后 开始 安装 ， 如 图 1.18 所 示 。 


(8) 安装 完成 后 可 以 在 “开始 ”菜单 的 AppServ 相关 操作 列表 中 启动 Apache 及 MySQL 服务 ， 
如 图 1.19 所 示 。 


区 7 Wi App5erv 2.5.7i5 beng nstaled， Completing the AppSery 2.5.7 Setup 


Wizard 


图 1.17 MySQL 设置 


Extract: server privieges,)s,,, 100% AppServ 2.5.7 has been Instaled on your computer, 
= 


[EF 


Chck Finich to dose this weard 


Fy Bt Apache 


Wi Start My5QL 


| 
ba 
[可 
a 
[站 
上 束 
| 人 
[2 


图 1.18 ”AppServ 安装 窗口 图 1.19 AppServ 安装 完成 窗口 


(9) 安装 好 AppServ 后 ， 整 个 目录 默认 安装 在 C:\AppServ 中 ， 此 目录 下 包含 4 个 子 目 录 ， 如 
图 1.20 所 示 ， 用 户 可 以 将 所 有 网 页 文件 存放 到 www 目录 下 。 


Apache 的 存储 目录 

IsqL 的 存储 目录 

PiE 的 存储 路 径 

由 站 ww 一 一 一 网 页 文件 及 phpllydnin 的 存储 路 径 


图 1.20 AppServ 目录 结构 


11 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


(10) 打开 浏览 器 , 在 地 址 栏 中 输入 “http:Wlocalhost” 或 者 “http:/127.0.0.1”， 如 果 打 开 如 图 1.21 
所 示 的 网 页 ， 则 说 明 AppServ 安装 成 功 。 


| XH 

[GR a le 

The AppServ Open Project - 2.5.7 for Windows 
ES 
E37 


phpMyAdmin Database Manager Version 2.9.0.2 
PHP Intormation Version 5.1.6 


AboutAppSev Version ?57 for Wndows 
AppSen a mercng open source sofware installer parkogeforWindomsincudes : 


» Apacho Web SarverVerscn223 


Change Longuage : El Si 


对 Easy way to build Webaerver, Database Server with AppServ :-) 前 
me a 


图 1.21 AppServ 测试 页 


1.4.2 Windows 下 Apache 的 安装 配置 


Apache 是 全 世界 使 用 范围 最 广 的 Web 服务 软件 ， 超 过 50% 的 网 站 都 在 使 用 Apache 服务 器 ， 它 以 
高 效 、 稳 定 、 安 全 、 免 费 而 成 为 了 最 受 欢 迎 的 服务 器 软件 。 

本 节 主 要 介绍 如 何在 Windows 操作 系统 中 安装 和 配置 Apache 服务 器 。 安 装 Apache 服务 器 前 ， 应 
到 官方 网 站 http:/www.apache.org 下 载 Apache 的 安装 程序 。 


鳃 和 视频 录像 : TM\01\x\Windows 下 Apache 的 安装 配置 .swf 


在 Windows 下 实现 Apache 的 安装 配置 的 操作 步骤 如 下 : 

(1) 下 载 Apache 的 安装 包 apache 2.0.59-win32-x86-no_sslmsi 后 ， 双 击 该 安装 包 ， 打 开 如 图 1.22 
所 示 的 Apache 的 安装 窗口 。 

(2) 在 图 1.22 中 单 击 Next 按钮 ， 打 开 如 图 1.23 所 示 的 Apache 许可 协议 窗口 ， 选 中 I accept the 
terms in the license agreement 单 选 按钮 。 

(3) 在 图 1.23 中 单 击 Next 按钮 ， 打 开 如 图 1.24 所 示 的 HTTP 服务 窗口 。 

(4) 在 图 1.24 中 单 击 Next 按钮 ， 打 开 如 图 1.25 所 示 的 输入 服务 器 信息 窗口 。 输 入 服务 器 的 相关 
信息 ， 如 网 络 域名 、 服 务 器 名 和 管理 员 邮 箱 等 ， 这 里 可 以 根据 用 户 的 实际 情况 输入 。 在 下 方 的 单 选 按 
钮 组 中 ， 如 果 选 择 第 一 项 则 可 以 对 任何 用 户 开 放 Apache 服务 ， 同 时 设置 服务 器 的 侦 听 端口 为 80， 如 
果 选 择 第 二 项 则 只 有 本 地 用 户 可 以 连接 和 使 用 Apache 服务 。 


全 注意 : 如 果 选 择 的 是 第 一 项 ， 又 同时 安装 了 IIS， 那 就 必须 修改 IIS 的 默认 端口 ， 否 则 将 导致 
Apache 无 法 正常 工作 。 更 改 IIS 的 默认 侦 听 端口 为 80， 可 以 在 IIS 的 管理 器 中 进行 设置 ， 
或 者 停止 IIS 的 服务 也 可 以 。 


了 
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pache HITP Server 2.0 


Installation Wizard 


‘Welcome to the Installation Wizard for 
Apache HTTP Server 2.0.59 


The Installation Wizard wil install Apache HTTP Server 2.0.59 
on your computer. To continue, cick Next. 


‘WARNING: This program is protected by copyright law and 
international treaties. 


License Agreement 


Apache License 
Version 2.0, January 2004 
http: /www.apache.org/licenses/ 


ERMS AND CONDITIONS FOR 
选择 接受 许可 协议 


1. Definitions, 


Installshield 


”4 Read thi Before Running Apache on Windows， 


cm [es co | 


图 1.23 


许可 协议 窗口 


Apache JITTP 


Read This First 


Apache HTTP Server 


le Apache HTTP Server is a powerful and flexible HTTP/1.1 compliant web server. 
lOriginally designed as a replacement for the NCSA HTTP Server, ithas grown to be 
[he most popular web server on the Internet. As a project of the Apache Software 
IFoundation, the developers aim to collaboratively develop and maintain a robust, 
Icommercial-grade, standards-based server with freely available source code. 


Detalls ofthe latest version can be found on the Apache HTTP server project page 
under 
htpyh 


Instalishiald 


es co | 


图 1.24 HTTP 服务 窗口 


Server Information 
Please enter your server's information. 
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Instal Apache HTTP Server 2.0 programs and shortcuts For: 


他 for A Users, on Port 80, as a Service -- Recommended. 
ny TorT Ter, on Port teq Wenaly, 


Instalshield 


cok es | cre 


图 1.25 输入 服务 器 信息 窗口 
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(5) 设置 完成 后 ， 单 击 图 1.25 中 的 Next 按钮 ， 打 开 如 图 1.26 所 示 的 选择 安装 方式 窗口 。 这 里 有 
两 种 选择 方式 ;Typical 典型 安装 和 Custom 自 定义 安装 。 通 常 ， 用 户 都 选择 典型 安装 方式 ， 单 
按钮 ， 打 开 如 图 1.27 所 示 的 设置 安装 路 径 窗口 。 


全 Apache MTTP Server 2.0 - Installatioa Wizare 


Setup Type 


Choose the setup type that bestsuits your needs, 


Please select a setup type, 


和 鲁 nstal Apache HTTP Server 2.0 to the folder: 
Ciprogramn Fies\Apache Group 


modules， 
选择 典型 安装 


Ta 


Tc 
图 1.26 选择 安装 方式 图 1.27 设置 安装 的 路 径 


(6) 在 如 图 1.27 所 示 的 设置 安装 的 路 径 窗 口中 可 以 选择 安装 的 路 径 ， 单 击 Change 按钮 ， 打 开 如 
图 1.28 所 示 的 对 话 框 来 修改 文件 的 安装 路 径 。 


(7) 在 如 图 1.28 所 示 的 窗口 中 ， 对 安装 的 路 径 进 行 修改 ， 然 后 单 击 OK 按钮 ， 打 开 如 图 1.29 所 
示 的 准备 安装 窗口 。 


(8) 在 图 1.29 中 单 击 Install 按钮 ， 打 开 如 图 1.30 所 示 的 窗口 开始 安装 。 


EPE Server 2 on Wi x 
Ready to Install the Program 
Sorver 2.0 — Tostelletio ye The ward s ready to begn Instalation. 
Browse to the destnaton folder, -df 


Chick Install to begin the nstalation 


bookin: 


[EF poche Go ， 习 加 吉 


1f you want to review or change any of your instalation settings, cjck Back Clck Cancel to 
exit the wizard, 


图 1.28 修改 安装 的 路 径 图 1.29 准备 安装 


(9) 图 1.30 中 所 有 文件 复制 完成 后 ，Apache 安装 完成 ， 打 开 如 图 1.31 所 示 的 窗口 ， 单 击 Finish 
按钮 完成 Apache 安装 。 
(10) 接 下 来 ， 测 试 Apache 服务 是 否 安装 成 功 。 选 择 “ 开 始 ”/“ 所 有 程序 ”命令 ， 在 弹出 的 菜 


单 中 能 够 看 到 Apache 服务 器 相关 操作 列表 ， 同 时 如 果 在 系统 托盘 中 有 一 个 策 图 标 ， 风 
务 已 经 启动 。 


表示 Apache 服 
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r 2.0 — Installation Vizard 


Installation Wizard Completed 


The program features you selected are being instaled. ”4 


_ 回 Please wait while the Instalation Wizard nstals Apache HTTP Server The Instalation Wizard has successfuly nstaled Apache HTTP 
”20:59 The may take several minutes. Server 2.0.59, Cidk Finish to et the Wizard. 

Status: 

Copying new fles 


LTT 


Instalishield 


图 1.30 复制 文件 图 1.31 安装 完成 


(11) 单 击 旺 图 标 后 ， 打 开 如 图 1.32 所 示 的 窗口 ， 在 该 窗口 中 ， 选 择 Stop 选项 表示 停止 Apache 
服务 器 ; 选择 Start 选项 表示 启动 Apache 服务 器 (Apache 服务 器 的 当前 状态 为 停止 时 ， 该 功能 可 用 ) ; 
选择 Restart 选项 表示 重新 启动 Apache 服务 器 。 双击 鸯 图 标 后 ， 打 开 如 图 1.33 所 示 的 窗口 ， 可 以 进行 
启动 和 停止 服务 器 的 各 种 操作 。 


启动 Apache 服务 器 


Start 
Stop 
Restart 


外 和 Apache2 上 


pache/2.2.3 OWin32) 
图 1.32 左 键 菜单 图 1.33 ”Apache 服务 器 的 监视 器 


(12) 在 浏览 器 中 输入 “http://localhost/” 或 者 “http://127.0.0.1/”， 如 果 能 够 浏览 到 如 图 1.34 所 
示 的 窗口 ， 说 明 Apache 服务 器 安装 成 功 。 

(13) Apache 服务 器 安装 成 功 后 ， 接 下 来 对 其 进行 配置 ， 以 便 Apache 服务 器 能 够 识别 PHP 文件 。 
配置 Apache 服务 器 主要 是 在 Apache 安装 目录 下 的 conf 子 目录 中 的 httpd.conf 文件 中 进行 的 ， 找 到 该 
文件 并 用 记事 本 等 文本 编辑 器 打开 该 文件 。 

(14) 定位 到 DocumentRoot 一 行 ， 可 以 将 该 路 径 修改 为 合适 的 路 径 ， 例 如 ， 设 置 为 DocumentRoot 
"C:\Apache\Apache2vwebpage" 表 示 Apache 服务 器 的 虚拟 目录 为 C:\Apache\Apache2\webpage。 

(15) 将 光标 定位 到 DirectoryIndex index.html index.html.var， 在 其 后 面 添加 一 个 PHP 默认 页 ， 通 
常 是 index.php， 表 示 当 访问 该 服务 器 时 如 果 未 指定 要 访问 的 PHP 文件 ， 则 默认 访问 index.php 文件 。 
更 改 后 的 代码 如 下 : 
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DirectoryIndex index.html index.html.var index.php 


[ano ET EE WO AD R00 
E 中 


这 不 是 你 想 看 见 的 页 面 吧 ? 


Mess 人 加 果 有 渗 问 ， 请 冰 询 维护 此 站 点 的 人 
ache 软 件 于 例会。 网 站 服务 拒 骏 社 的 开发 者 ， 不 名 黄 此 站点 的 绒 护 工作 ， 也 无 法 
ne 


Apache 文档 忆 经 旭 合 在 此 发 行 舌 中 。 
您 可 以 在 使 用 Apache 的 网 站 服务 如 上 ， 自 由 地 使 用 下 面 的 图 片 。 夺 副 便 用 如 achel1 


Ta 一 一 一 上 


I 玫 吉 eaet 
图 1.34 测试 Apache 服务 器 是 否 安装 成 功 
(16) 为 了 使 Apache 能 够 识别 PHP 的 扩展 名 ， 搜 索 并 定位 到 httpd.conf 文件 的 如 下 部 分 : 


<Directory "C:/Apache/Apache2/cgi-bin"> 
AllowOverride None 
Options None 
Order allow,deny 
Allow from all 
</Directory> 


在 后 面 添 加 如 下 两 行 代码 : 


AddType application/x-httpd-php .php .phtml .php3 .php4 
AddType application/x-httpd-php-source .phps 


(17) 为 了 能 够 使 用 模块 功能 ， 模 块 通常 以 DSO 方式 构建 ， 读 者 需要 定位 到 如 下 代码 ， 
#LoadModule ssl module modules/mod ssl.so 


使 得 能 够 在 使 用 前 获得 指令 的 功能 ， 然 后 以 module 方式 加 载 PHP， 指 向 PHP 5.0 目录 下 的 
php5apache2.dll 文件 ， 添 加 如 下 代码 : 


LoadModule php5_module c:\phps\phpsapache2.dll 


wll 


1.4.3 Windows 下 MySQL 的 安装 配置 


PHP 能 够 支持 市 面 上 的 大 多 数 数据 库 ， 如 MySQL、Access、SQL Server、Oracle 等 ， 而 MySQL 
数据 库 是 PHP 程序 开发 人 员 公认 的 黄金 搭档 ， 这 不 仅 因为 MySQL 是 完全 网 络 化 的 跨 平台 关系 型 数据 
库 系 统 ， 也 是 具有 客户 机 /服务 器 体系 结构 的 分 布 式 数据 库 管理 系统 。 它 具有 功能 性 强 、 使 用 简洁 、 管 
理 方便 、 运 行 速度 快 、 版 本 升级 快 、 安 全 性 高 等 优点 ， 而 且 MySQL 数据 库 完全 免费 ， 从 官方 网 站 
http://www.mysql.com 即 可 免费 下 载 到 最 新 版 本 的 MySQL 安装 包 。 
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鳃 和 视频 录像 : TMNOINIXVWindows 下 MySQL 的 安装 配置 swf 


在 Windows 下 实现 MySQL 的 安装 配置 的 操作 步骤 如 下 : 

(1) 如 果 下 载 的 是 Windows Essentials (x86) ， 例 如 mysql-4.1.11-essential-win.exe， 直 接 双击 该 
文件 安装 即 可 ， 安 装 完毕 后 ， 在 “开始 ”菜单 中 选择 “运行 ”命令 ， 在 弹出 的 对 话 框 中 输入 cmd， 进 
入 “命令 提示 符 ”， 输 入 mysqld-nt.exe， 即 可 启动 MySQL 服务 ;如 果 还 想 详细 地 配置 MySQL， 则 需 
要 进入 Ci\mysql\bin 目录 ， 运 行 MySQLInstanceConfig.exe， 并 按 提示 操作 即 可 。 

(2) 如 果 下 载 的 是 Windows (x86) ， 例 如 mysql-essential-5.0.24-win32.msi， 解 压 该 文件 后 双击 
setup.exe 安装 程序 即 可 逐步 完成 MySQL 的 安装 。 

(3) 如 果 下 载 的 是 Without noinstall (unzip in C:\) ， 直 接 将 该 文件 解压 到 指定 的 安装 目录 下 。 在 

“开始 ”菜单 中 选择 “运行 ”命令 ， 在 弹出 的 对 话 框 中 输入 cmd， 然 后 进入 “命令 提示 符 ”， 输 入 
mysqld-nt.exe 命令 ， 即 可 启动 MySQL 服务 器 。 


1.4.4 Windows 下 PHP 的 安装 配置 


架设 基于 PHP 的 Web 服务 器 , 安装 PHP 是 必须 的 。 由 于 PHP 的 代码 公开 , 所 以 其 升级 速度 较 快 。 
安装 PHP 之 前 应 从 官方 网 站 http://www.php.net/ 下 载 最 新 版 本 的 PHP 安装 程序 ， 下 面 以 
php-5.1.4-Win32.zip 为 例 讲 解 PHP 的 配置 方法 。 


鳃 视频 录像 : 光盘 TM\01\x\Windows 下 PHP 的 安装 配置 .swf 


在 Windows 下 实现 PHP 的 安装 配置 的 操作 步骤 如 下 : 
(1) 成 功 下 载 PHP 的 安装 包 后 ， 首 先 应 对 其 进行 解压 ， 如 解压 到 C:\php5 目录 下 ， 如 图 1.35 所 示 。 


ET TT | 和 
JO- 小 让 扫 记 文 人 | 他 
Ew [De ws 


文件 和 文件 实 任 务 < 
局 TWH 
@ TNR 


局 共 训 kxh 天 
其 它们 村 
BME a 
te BA: zoorFaAii 
人 和 

后 1 率 


1.35 PHP 解压 后 的 文件 


(2) 将 Ci\php5\ 目 录 下 的 所 有 .dll 文件 复制 到 C:\Windows\system32\〈 如 果 是 Windows 2000 操作 
系统 ， 则 为 CVWINNT\system32\) 目录 下 。 

(3) 将 C:\phps\ 目 录 下 的 php.ini-dist 文件 复制 到 C:\Windows\〔 如 果 是 Windows 2000 操作 系统 ， 
则 为 C:\WINNT\) 目录 下 ， 然 后 将 php.ini-dist 重 命名 为 php.ini， 并 用 记事 本 打开 该 文件 进行 编辑 ， 具 
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体 编辑 方式 如 下 : 
首先 ， 定 位 到 该 文件 register_globals = O 仓 所 在 行 ， 将 O 企 改 为 On。 代码 如 下 : 
Tegister globals = On 
其 次 ， 定 位 到 该 文件 extension_dir="." 所 在 行 ， 将 路 径 改 为 "C:\php5\ext"。 代 码 如 下 : 
extension_dir="C:\php5\ext" 
最 后 ， 查 找 定位 Windows Extensions， 将 下 面 两 行 代码 : 
; extension=php_gd2.dll 
; extension=php_mysql.dll 
前 面 的 注释 〈 即 分 号 标识 符 ) 去掉， 这 样 PHP 就 可 以 支持 GD2 函数 库 和 MySQL 相关 函数 ， 如 果 

想 让 PHP 支持 其 他 功能 函数 ， 同 样 需 要 将 与 这 些 操作 函数 相关 的 dl 文件 前 的 注释 去 掉 。 

全 注意 ; 以 上 为 PHP 的 常规 配置 ， 在 PHP 的 官方 论坛 上 可 以 查 到 更 多 的 详细 配置 。 由 于 PHP 的 版 
本 有 多 种 ， 所 以 php.ini 的 配置 也 存在 差别 ， 具 体 使 用 时 应 根据 实际 情况 配置 php.ini 文件 ， 
php.ini 文件 的 配置 信息 在 2.5.2 节 有 详细 介绍 。 

(4) PHP 配置 完成 以 后 ， 需 要 重新 启动 Apache 服务 器 ， 然 后 编写 test.php 文件 并 输入 如 下 代码 : 


<2php 
phpinfo0: 
?> 


将 文件 保存 到 Apache 安装 目录 下 的 webpage 子 目录 下 , 然后 在 浏览 器 中 输入 http:/127.0.0.l/testphp， 
如 果 显 示 如 图 1.36 所 示 的 页 面 则 说 明 PHP 配置 成 功 。 


加 | 相 


B® 
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图 1.36 测试 成 功 
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总 技巧: 安装 文件 的 路 径 也 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移 植 程序 时 带 来 
的 不 便 ， 选 择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹配 。 建 议 最 好 不 要 选 
择 中 间 有 空格 的 目录 ， 如 E:\program Files\PHP， 这 样 做 会 导致 一 些 未 知 错误 或 崩 演 发 生 。 


1.4.5 在 Windows 下 设置 IIS+PHP 的 执行 环境 


PHP 可 以 在 很 多 服务 器 下 运行 ， 下 面 介绍 IIS 的 安装 方法 以 及 PHP 是 如 何在 IIS 服务 器 下 运行 的 。 

1. lIS 的 安装 

Internet 信息 服务 器 (Internet Information Server， 缩 写 为 IS) 是 Microsoft 的 Web 服务 器 ， 它 集成 
于 Windows NT Server 之 中 ， 方 便 易 用 ， 为 Web 应 用 程序 提供 了 功能 强大 的 运行 平台 。 

IIS 目前 最 新 版 本 为 6.0, 在 Windows 2003 Server 操 作 系统 中 默认 安装 , 如 果 没 有 安装 Windows 2003 
Server， 可 以 选择 IS 5.0， 它 可 以 在 Windows XP 或 者 Windows 2000 操作 系统 中 安装 ， 对 于 开发 而 言 ， 
IIS 的 各 个 版 本 之 间 的 区 别 并 不 大 。 

(1) 在 Windows XP 或 者 Windows 2000 操作 系统 下 安装 IS。 
名 0 视频 录 像 :TM\O1\Ix\ 在 Windows XP 或 者 Windows 2000 操作 系统 下 安装 IIS.swf 


选择 “开始 ”/“ 设 置 ”/“ 控 制 面板 ”命令 ， 启 动 “添加 /删除 程序 ”应 用 程序 ， 单 击 “ 添 加 /删除 
Windows 组 件 ” 图 标 ， 在 弹出 的 “Windows 组 件 向 导 ” 对 话 框 的 “组 件 ” 列 表 框 中 选择 “Internet 信息 
服务 (IIS) ”选项 ， 最 后 单 击 “ 下 一 步 ” 按 钮 ， 即 可 完成 IS 的 安装 。 
(2) 在 Windows 2003 Server 服务 器 下 安装 IIS。 


鳃 和 视频 录像 : TMNOINIX\ 在 Windows 2003 Server 服务 器 下 安装 IS.swf 


首先 选择 “开始 ”/“ 控 制 面板 ”命令 ,然后 启动 “添加 /删除 程序 ”应 用 程序 ， 单 击 “ 添 加 /删除 
Windows 组 件 ” 图 标 ， 在 弹出 的 “Windows 组 件 向 导 ” 对 话 框 的 “组 件 ” 列 表 框 中 选择 “应 用 程序 服 
务 器 ”选项 ， 然 后 单 击 “ 详 细 信 息 ” 按 钮 ， 选 择 “Internet 信息 服务 (IIS) ”选项 ， 最 后 单 击 “ 下 一 步 ” 
按钮 ， 即 可 完成 HS 的 安装 。 


全 注意 : 若 想 在 IIS 中 支持 PHP 的 运行 环境 ， 在 安装 IIS 时 ， 需 在 “网 站 访问 权限 ”窗口 中 选中 读 
取 、 运 行 脚 本 (如 ASP) 、 执 行 (如 ISAPI 应 用 程序 或 CGI) 选项 。 
2. 架设 PHP 到 IIS 服务 器 
在 架设 PHP 到 IS 服务 器 之 前 , 首先 应 确保 PHP 已 经 正确 安装 到 系统 中 。 本 节 将 以 Windows 2003 
Server 服务 器 中 的 IS 6.0 为 例 ， 介 绍 架 设 PHP 到 IIS 服务 器 的 方法 。 
鳃 和 视频 录像 : 光盘 TM\01\Ix\ 在 Windows 下 设置 IS+PHP 的 执行 环境 .swf 
(1) 选择 “开始 ”/“ 管 理工 具 ”/“Intemet 信息 服务 (IS) 管理 器 ”命令 ， 打 开 如 图 1.37 所 示 


的 Intemet 信息 服务 窗口 。 
(2) 右 击 “默认 网 站 ”节点 ， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 将 打开 如 图 1.38 所 示 的 对 
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话 框 。 


图 1.38 默认 Web 站 点 属性 


(3) 单 击 图 1.38 中 “ 主 目录 ”选项 卡 下 的 “配置 ”按钮 ， 打 开 如 图 1.39 所 示 的 对 话 框 。 

(4) 单 击 图 1.39 中 的 “添加 ”按钮 将 打开 如 图 1.40 所 示 的 对 话 框 ， 单 击 “可 执行 文件 ”文本 杠 
后 的 “浏览 ”按钮 ， 在 打开 的 窗口 中 选择 phps 目录 下 的 phpsisapidll 文件 ， 在 “扩展 名 ”文本 框 中 输 
入 “php”， 然 后 单 击 “ 确 定 ”按钮 ， 实 现 扩展 名 的 映射 。 

(5) 单 击 “ 主 目录 ”选项 卡 中 的 “浏览 ”按钮 ， 选 择 网 站 的 路 径 ， 如 图 1.41 所 示 。 

(6) 在 “Intemet 信息 服务 (TS)》 管理 器 ”窗口 中 单 击 “Web 服务 扩展 ”节点 ， 在 “Web 服务 扩 
展 ”区 域 中 选择 “所 有 未 知 ISAPI 扩展 ”选项 ， 然 后 单 击 “允许 ” 按钮， 将 该 服务 的 状况 由 “禁止 ” 


设置 为 “允许 ”， 如 图 1.42 所 示 。 


Bs 
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MINDONS\systen32\inetsrv\e .. 
C: MINDOYWS\systen32\inetsrv\e. .. 
C: MINDOWS\systen32\inetsrv\a .. 
C: ‘FTNDOYS\ systen32\i! 由 
FAWTITWSYswstem32\i 


添加 /篇 辑 应 用 程 订 扩 展 名 映射 


图 1.42 设置 Web 服务 扩展 
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(7) 打开 IS 服务 器 ， 浏 览 mdex.php 文件 ， 如 果 在 浏览 器 中 打开 网 页 文件 ， 则 说 明 IS+PHP 运 
行 环 境 配置 成 功 ， 如 图 1.43 所 示 。 


| 让 诅 -四 - 口 国 全 | 万 抽 六 天 二 报 体 侣 |- 马 加 -二流 
WE 


TW 


图 1.43 IIS+PHP 运行 环境 配置 成 功 
1.5 ”数据 库 设计 
1.5.1 数据库 分 析 


本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 到 海量 数据 ， 因 此 需要 充分 考虑 到 成 
本 问题 及 用 于 需求 〈 如 跨 平 台 ) 等 问题 ， 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 
网 络 化 、 跨 平台 的 关系 型 数据 库 系 统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数 
据 库 。 


1.5.2 数据库 概 念 设 计 


根据 前 面 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实 体 分 别 为 免费 信息 
实体 、 付 费 信息 实体 、 广 告 信息 实体 、 管 理 员 实体 。 下 面 将 介绍 这 几 个 实体 的 E-R 图 。 

1， 免 费 信 息 实 体 

免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 审 核 状态 和 发 布 
时 间 属 性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 免 费 信 
息 实 体 的 E-R 图 如 图 1.44 所 示 。 

2. 付费 信息 实体 

付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 截 止 
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时 间 和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 否 ”。 
付费 信息 实体 的 E-R 图 如 图 1.45 所 示 。 


1.45 ”付费 信息 实体 E-R 
3. 广告 信息 实体 


广告 信息 实体 包括 编号 、 信 息 标 题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 来 
标识 信息 是 否 在 前 台 显示 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 否 ”。 广告 信息 实体 的 E-R 图 如 图 1.46 所 示 。 


图 1.46 广告 信息 实体 E-R 图 
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图 1.47 管理 员 实体 E-R 


1.5.3 ”使 用 PowerDesigner 建 模 


在 数据 库 概念 设计 中 已 经 分 析 了 本 系统 中 主要 的 数据 实体 对 象 ， 通 过 这 些 实体 可 以 得 出 数据 表 结 
构 的 基本 模型 ， 最 终 实 施 到 数据 库 中 ， 形 成 完整 的 数据 结构 。 下 面 将 介绍 使 用 PowerDesigner 工具 完成 
本 系统 的 数据 库 建 模 。 

(1) 运行 PowerDesigner， 并 在 PowerDesigner 主 窗口 中 选择 File/New 命令 ， 在 打开 的 New 对 话 
框 中 选择 Physical Data Model 物理 数据 模型 ， 简 称 PDM) 列表 项 ， 单 击 OK 按钮 ， 打 开 如 图 1.48 所 
示 的 Choose DBMS【〈 选 择 数据 库 管理 系统 ) 对 话 框 。 

EE | 
[> | | 
他 Shane Use the shared DBMS definiion 
三 Copy Create a copy of the DBMS defirition in model 


Ce | 
图 1.48 Choose DBMS 对 话 框 


(2) 在 DBMS 下 拉 列 表 框 中 选择 MySQL 数据 库 ， 这 里 选择 MySQL 3.23 版 本 ， 这 是 由 于 当前 使 
用 的 PowerDesigner 的 版 本 还 不 支持 MySQL 5.0， 但 是 随 着 PowerDesigner 版 本 的 升级 将 支持 更 多 的 
DBMS。 

(3) 其 他 选项 采用 默认 设置 即 可 ， 单 击 OK 按钮 ， 打 开 新 建 的 PDM 窗口 。 在 该 窗口 的 上 方 为 空 
的 图 形 窗口 ， 下 方 为 输出 窗口 。 其 中 图 形 窗口 的 右 侧 有 一 个 工具 面板 ， 如 图 1.49 所 示 。 

(4) 在 图 1.49 中 单 击 “ 建 立 表 ” 图 标 ,这 时 鼠标 指针 将 显示 为 日 ， 在 图 形 窗口 的 合适 位 置 单 击 鼠 
标 左 键 ， 此 时 在 图 形 窗口 中 将 显示 如 图 1.50 所 示 的 表 符 号 。 


全 注意: 细心 的 读者 可 以 发 现 ， 此 时 的 鼠标 指针 仍然 是 白 ， 如 果 再 单 击 鼠 标 左 键 还 将 出 现 类 似 图 1.50 
所 示 的 表 符 号 ， 如 果 想 取消 该 指针 ， 可 以 单 击 工具 面板 中 的 指针 图 标 险 。 


@ 
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Table 1 


图 1.49 工具 面板 图 1.50 表 符号 


(5) 在 图 1.50 所 示 的 表 符 号 上 双击 鼠标 ， 将 打开 Table Properties〈 表 属性 ) 对 话 框 ， 默认 情况 下 
在 该 对 话 框 中 选择 的 是 General 选项 卡 , 在 该 选项 卡 的 Name 文本 框 中 输入 表 的 名 字 tb_admin, 此 时 在 
Code 文 本 框 中 也 将 自动 显示 tb_admin， 其 他 选项 选择 默认 即 可 。 

(6) 选择 Columns 选项 卡 ， 首 先 单 击 列 输入 列表 的 第 一 行 ， 将 自动 填写 一 行 信息 ， 然 后 将 Name 
列 修 改 为 4， 同时 Code 列 也 将 自动 显示 为 id， 再 在 Data Type 列 中 选择 integer unsigned 列表 项 ， 最 后 
选中 了 P 列 的 复 选 框 ， 此 时 M 列 的 复 选 框 也 将 自动 选中 。 

(7) 按照 步骤 (6) 的 方法 再 添加 两 个 列 name 和 pwd， 如 图 1.51 所 示 。 

(8) 在 图 1.51 中 单 击 “ 应 用 ”按钮 后 ， 双 击 id 列 的 内 容 ， 将 打开 Column Properties( 列 属性 》 
对 话 框 ， 如 图 1.52 所 示 。 默 认 选择 General 选项 卡 ， 在 该 选项 卡 中 ， 选 中 Identity 复 选 框 ， 此 项 操作 用 
于 设置 id 列 为 自动 编号 列 。 


Notes | BRules 一 Dependencies | VYersion Info 


9 Generd | Detail | Standard checks | Additional Checks 
Wapping | Notes | Rules | Dependencies | Extended Dependencies | Version Info | 
General Columns | Indexes | Keys | check | Seript | Options | revie Means 
启 |' 轨 是 车 | 业 昌 写 X| 约 | 又 民 
[| ce | 


Table: [Ex——| 
Daatpe fr | Diplyed 


tk Becson 站 
Domsm [EX ” 司 贺 


BPimaykey FF Eoeonkey 了 区 Mandatoy 


图 1.51 Columns 选项 卡 1.52 ”Columns 列 属性 对 话 框 
(9) 单 击 “ 应 用 ”按钮 后 ， 再 单 击 “ 确 定 ” 按 钮 ， 关 闭 Column Properties 对 话 框 。 


(10) 单 击 “ 确 定 ” 按 钮 ， 关 闭 Table Properties 对 话 框 ， 完 成 tb_admin 表 的 创建 。 
(11) 按照 步骤 (4) 一 步骤 (10) 的 方法 创建 本 系统 中 的 其 他 数据 表 。 创建 完成 的 模型 图 如 图 1.53 
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所 示 。 


th admin 


th advertising 


th_leaguerinfo 


id int 


id 


name varchar (50) 
pwd varchar (50) 


Spk> | | tite 


fdate datetime 


flag int() 


varchar (100) 
content varchar(500) 


tb_info 
id 

这 int pl | | type 

plo | type varchar (30) title 
title varchar (50) content 
content varchar (500) Te 
linkman varchar (20) tel 
tel varchar (30) sdate 
checkstate int(1) showday 
edate datetime checkstate 


int p> 
varchar (20) 
varchar (50) 
varchar (500) 
varchar (20) 
varchar (30) 

date 

date 

int (1) 


图 1.53 九 九 度 供求 信息 网 的 模型 图 


从 技巧 ， 在 默认 情况 下 ， 创 建 后 的 表 符号 中 的 全 部 文字 均 为 常规 样式 的 宋体 8 号 字 ， 如 果 想 修改 广 


字 的 格式 ， 可 以 选中 全 部 表 符 号 ， 按 CtrlHT 快捷 键 ， 在 打开 的 Symbol Format 对 话 框 中 选 
择 Font 选项 卡 ， 在 该 选项 卡 中 设置 相关 内 容 的 字体 及 样式 和 字号 等 。 
(12) 选择 PowerDesigner/Database/Generate Database 命令 ， 将 打开 Database Generation 对 话 框 ， 


在 该 对 话 框 中 设置 导出 的 脚本 文件 的 名 称 〈 如 pursey.sql) 及 保存 路 径 〈 如 F'\pursey) ， 在 下 方 的 各 个 
选项 卡 中 设置 相关 的 导出 内 容 ， 设 置 完毕 后 ， 单 击 “ 确 定 ”按钮 ， 导 出 数据 库 脚 本 文件 完成 。 


上 说 明 : 通过 以 上 方法 导出 的 脚本 文件 ， 可 以 在 MySQL 的 客户 端 命令 行 窗口 中 通过 source 命令 执 


行 ， 并 创建 相应 的 数据 表 。 


1.5.4 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需 求 的 分 析 , 可 知 九 九 度 供求 信息 网 中 应 用 的 db_pursey 数据 库 主要 包含 如 
下 4 个 数据 表 ， 如 图 1.54 所 示 。 


胃 服务 器 : localhost 》 暂 数据 库 : db_pursey 
表 类 型 整理 说 明 
tb_admin MyISAM gb2312_chinese_ci 。 管理 员 信息 表 
tb_advertising ”MyISAM ”gb2312_chinese_ci 。 企业 广告 信息 表 
th_info MyISAM ”gb2312_chinese_ci 。 免费 供求 信息 表 
tb_leaguerinfo MyISAM gb2312_chinese_ci 。 付费 供求 信息 素 
图 1.54 九 九 度 供求 信息 网 数据 表 


各 数据 表 的 表 结 构 如 图 1.55 一 图 1.58 所 示 。 
1. tb_admin (管理 员 信 息 表 ) 
管理 员 信息 表 主 要 用 于 存储 管理 员 的 信息 。 该 数据 表 的 结构 如 图 1.55 所 示 。 


胃 服务 器 : localhost ， 电 数据 库 : db_pursey， 国 表 : tb_admin 
突 外 


字段 
过 int(4) 


Null 于 认 
理 


mame varchar(50) gb2312_chinese_ci 是 NULL 
pwd 。 varchar(50) gb2312_chinese_ci 是 WULL 


1.55 


说 明 
auto_increment ”自动 编号 


管理 员 信息 表 结 构 
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2. tb_advertising (企业 广告 信息 表 ) 
企业 广告 信息 表 主 要 用 于 存储 企业 发 布 的 广告 信息 。 该 数据 表 的 结构 如 图 1.56 所 示 。 


加 服务 器 : localhost > 次 数据 库 : db_pursey， 国 表 : tb_advertising 
字 稻 类 型 整理 Null 。 著 认 额外 


到 int(4) 否 auto_increment 。 自动 编号 
title varchar(100) gb2312_chinese_ci 否 广告 标 是 
content 。 varchar(500) gb2312_chinese_ci 否 广告 内 容 
fdate datetime 否 发 布 时 间 
flag int(t) 否 0 推荐 状态 


图 1.56 企业 广告 信息 表 结构 
3. tb_info〔〈 免 费 供求 信息 表 ) 
免费 供求 信息 表 主 要 用 于 存储 用 户 免费 发 布 的 供求 信息 。 该 数据 表 的 结构 如 图 1.57 所 示 。 


困 服务 器 : localhost 加 数据 库 : db_pursey ， 图 表 :tb_info 
字段 类 型 Null 默认 额外 
int(4) auto_increment 
varchar(30) 。 gb2312_chinese_cl 
varchar(50) 。 gb2312_chinese_el 


varchar(500) gb2312_chinese_ci 
varchar(20) gb2312_chinese_ci 
Yarchar(30) gb2312_chinese_ci 
int(1) 

datetime 


现 叫 叫 叫 珊 叫 呀 叫 


1.57 ”免费 供求 信息 表 结 构 
4. tb_leaguerinfo 〈 付 费 供 求 信息 表 ) 
付费 供求 信息 表 主 要 用 于 存储 付费 的 供求 信息 。 该 数据 表 的 结构 如 图 1.58 所 示 。 


加 服务 器 : localhost 加 数据 库 : db_pursey ， 图 表 :tb_leaguerinfo 
字段 类 型 Null 额外 说 明 

这 int(4) 和 理 auto_increment “自动 编号 
type varchar(20) 。 gb2312_chinese_ci 否 信息 类 型 
title varchar(50) 。 gb2312_chinese_cl 否 信息 标题 
content varchar(500) ”gb2312_chinese_ci 否 信息 内 容 
linkman varchar(20) ”gb2312_chinese_ci 否 联系 人 
tel varchar(30) 。 gb2312_chinese_ci 否 联系 电话 
sdate date 理 发 布 日 期 
showday date 否 截止 日 其 
checkstate int(1) 否 0 审核 状态 


图 1.58 付费 供求 信息 表 结构 
1.6 单元 测试 


在 现代 软件 开发 过 程 中 ， 测 试 不 再 作为 一 个 独立 的 生命 周期 ， 而 是 成 为 与 编写 代码 同步 进行 的 开 
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发 活动 。 单 元 测试 能 够 提高 程序 员 对 程序 的 信心 ， 保 证 程序 的 质量 ， 加 快 软件 开发 速度 ， 使 程序 易于 
维护 。 


1.6.1 单元 测试 概述 


在 程序 设计 过 程 中 会 有 多 种 测试 ， 单 元 测试 只 是 其 中 的 一 种 。 单 元 测试 并 不 能 保证 程序 是 完美 无 
缺 的 ， 但 是 在 所 有 的 测试 中 ， 单 元 测试 是 第 一 个 环节 ， 也 是 最 重要 的 一 个 环节 。 单 元 测试 是 一 种 由 程 
序 员 自 行 测试 的 工作 。 简 单 地 说 ， 单 元 测试 就 是 测试 代码 撰写 者 依据 其 所 设想 的 方式 执行 是 否 产生 了 
预期 的 结果 。 

单元 测试 不 仅 是 无 错 编码 的 一 种 辅助 手段 ， 还 必须 是 可 重复 的 ， 即 无 论 是 在 软件 修改 或 是 移植 到 
新 的 运行 环境 的 过 程 中 都 可 以 进行 单元 测试 。 

与 单元 测试 有 密切 关联 的 开发 活动 包括 代码 走读 〈Code review) 、 静 态 分 析 (Static analysis) 和 
动态 分 析 (Dynamic analysis) 。 代 码 走读 就 是 对 软件 的 源 代码 进 行 宏观 地 阅读 ， 整 理 开 发 思路 。 静 态 
分 析 就 是 对 软件 的 源 代码 进行 研读 ， 查 找 错误 或 收集 一 些 度量 数据 ， 并 不 需要 对 代码 进行 编译 和 执行 。 
动态 分 析 就 是 通过 观察 软件 运行 时 的 动作 ， 来 提供 执行 跟踪 、 时 间 分 析 以 及 测试 覆盖 度 方面 的 信息 。 


1.6.2 ”单元 测试 的 优点 


单元 测试 具有 以 下 优点 : 

回 ”一 种 验证 行为 。 程 序 中 的 每 一 项 功能 都 是 用 测试 来 验证 它 的 正确 性 ， 为 以 后 的 开发 提供 支援 。 
就 算是 开发 后 期 ， 也 可 以 轻松 地 增加 功能 或 更 改 程序 结构 ， 而 不 用 担心 这 个 过 程 中 会 破坏 重 
要 的 东西 ， 而 且 它 为 代码 的 重 构 提供 了 保障 。 这 样 ， 我 们 就 可 以 更 自由 地 对 程序 进行 改进 。 

回 ”一 种 设计 行为 。 编 写 单元 测试 将 使 用 户 从 调用 者 观察 、 思 考 。 特 别 是 先 写 测试 (test-first》， 
人 迫使 设计 者 把 程序 设计 成 易于 调用 和 可 测试 的 ， 即 解除 软件 中 的 耦合 。 

加 一 种 编写 文档 的 行为 。 单 元 测试 是 一 种 无 价 的 文档 , 它 是 展示 函数 或 类 如 何 使 用 的 最 佳 文档 。 
这 份 文档 是 可 编译 、 可 运行 的 ， 并 且 它 保持 最 新 ， 永 远 与 代码 同步 。 

回 具有 回归 性 。 自 动 化 的 单元 测试 避免 了 代码 出 现 回归 ， 编 写 完成 之 后 ， 可 以 随时 随地 地 快速 
运行 测试 。 


1.7.1 前 台 首 页 概述 


网 站 主页 是 关于 网 站 的 建设 及 形象 宣传 ， 它 对 网 站 生存 和 发 展 起 着 非常 重要 的 作用 ， 应 该 是 一 个 
信息 含量 较 高 、 内 容 较 丰富 的 宣传 平台 。 九 九 度 供求 信息 网 前 全 首页 主要 包含 以 下 内 容 : 
回 ”网 站 菜单 导航 〈 包 括 公 寓 信 息 、 招 聘 信 息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信息 、 车 
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回 


辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 等 ) 。 

发 布 免费 的 供求 信息 包括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信 息 、 招 商 引 资 、 寻 人 / 物 启示 等 )。 

推荐 供求 信息 显示 〔 包 括 公 寓 信 息 、 招 聘 信息 、 求 职 信息 、 培 训 信 息 、 家 教 信息 、 房 屋 信息 、 
车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启 示 等 ) ， 其 中 ， 付 费 信息 按时 间 顺 序 
降序 排列 ， 免 费 信息 按时 间 顺 序 分 页 显示 。 

显示 推荐 的 企业 广告 信息 。 

供求 信息 快速 检索 : 支持 模糊 查询 和 查询 关键 字 描 红 功能 。 

后 台 登 录入 口 : 为 管理 员 进 入 后 台 提 供 一 个 入 口 。 


下 面 看 一 下 本 案例 中 提供 的 前 台 首 页 ， 如 图 1.59 所 示 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\01 
\99pursey\index.php。 


[2 

位 于 市 中 心 ,4 起 了 用 面 t 内 设 四 人 间 ,五 人 间 新 购置 标准 平 铺 单 人 床 , 新 被 独 床 单 。 单 独 
客厅 ,方便 起 居 。 有 厂 电 视 , 新 用 千 精 ,衣柜 ,床头柜 ， 柱 凡 # 可 做 白 ， eR 内 往 均 
为 有 相对 稻 定 工作 的 女孩 # 享 内 温 酚 ， 安放， 考 人 管理 ? 是 高 索 质 白领 女性 的 温 心 家 | 


信息 检索 区 二 可 和 人 1 本 办 士 。 可不 几率 1 Tagsszrpy 


加 信息 快速 检索 


全 吝 可 晤 】 4r 交 于 会 填 2007-12-20 0 和 
本 全 寅 位 于 舍 阳 街 , 交通 便利 ， 环境 

促 女 孩 入 住 ,有 不 度 哮 好 者 油 拢 。 

可 世人 MMR 文士 了 其 昌 湛 : 00l-anrzones 

[i 09 /入 : 乱 
民 庙 路 附近 , 交通 便利 ， 内 设 热水器 可 以 页 饭 ， 过 去 机 ,饮水 机 ， 电视 机 ,24 小 时 冷 热 水 

等 一 供 咀 好 ,全 新 被 竺 ， 客 扫 明 高 环境 南非 清 祝 "安全 卫生 ， 饮 碟 欢 迎 各 界 高 索 质 单身 男士 

音素 入 住 


了 和 人 本 和 主 。 取 系 遇 汪 7 Lose 
全 高 喜 外 外 五 度 关 于 入 评 2077-12-00 09:33: 别 
温 蛋 竺 附近 , 2 从 ,提供 2 人 间 , 8 人 间 。 交 通 舍利 ,室内 千 净 ， 伏 洁 ， 通风 好 ,有 用 千 阐 。 


有 电视 * 扣 供 热 水 。 价格 合理 , 亲生 化 考 人 管理 ,是 理 下 的 公 宣 进 择 。 
联系 人 = 王 卫 芒 。 联 每 员 所 :6653939 


www.bc110.cofm 
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区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 1.3 所 示 。 
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表 1.3 页 面 框 架 中 各 区 域 介绍 及 对 应 PHP 文件 


说 ”了 明 
提供 查看 各 类 信息 的 超 链接 
企业 广告 显示 及 各 类 信息 检索 
根据 用 户 请 求 显示 相应 内 容 

显示 版 权 信 息 


对 应 PHP 文 件 


left php 
根据 请 求 加 载 相应 的 PHP 文件 ， 默 认 加 载 main.php 


bottom | 


信息 检索 区 
内 容 显示 区 
版 权 区 


1.7.2 前 台 首 页 技术 分 析 


超 链接 在 本 质 上 属于 一 个 网 页 的 一 部 分 ， 它 是 一 种 允许 用 户 同 其 他 网 页 或 站 点 之 间 进 行 连接 的 元 
素 。 各 个 网 页 链接 在 一 起 后 ， 才 能 真正 构成 一 个 网 站 。 按 照 使 用 对 象 的 不 同 ， 网 页 中 的 链接 可 以 分 为 
文本 超 链接 、 图 像 超 链接 、E-mail 链接 、 锚 点 链接 、 多 媒体 文件 链接 和 空 链接 等 。 

图 像 不 但 可 以 建立 超 链接 ， 还 可 以 实现 图 像 映射 。 图 像 映射 是 指 一 幅 图 像 可 以 建立 多 个 超 链接 ， 
即 在 图 像 上 定义 多 个 区 域 , 每 个 区 域 链接 到 不 同 的 地 址 , 这 样 的 区 域 称 为 热 区 (也 称 为 热点 , Hotspot) 。 
本 项 目 就 是 应 用 图 像 热 区 实现 的 功能 导航 。 

图 像 映 射 有 两 种 : 服务 器 端 映射 (Server-side Image Map) 和 客户 端 映 射 (Client-side Image Map) 。 
目前 使 用 最 多 的 是 客户 端 映射 , 因为 客户 端 映射 使 图 像 上 对 应 区 域 的 坐标 以 及 链接 的 URL 地 址 都 在 浏 
览 器 端 读 入 ， 省 去 和 服务 器 之 间 互 传 坐标 和 URL 的 时 间 。 

在 PHP 文件 中 ， 使 用 <MAP> 标 记 创建 图 像 映射 。 语 法 如 下 : 

<IMG SRC= “file name” USEMAP= “#MapName” > 

<MAP NAME= “MapName” > 

<AREA SHAPE= “value” COORDS= “坐标 ” HREF=“URL” ALT= “描述 文字 ”> 
<AREA SHAPE= “value” COORDS= “坐标 ” HREF=“URL” ALT=“ 描 述 文字 ”> 


</MAP> 
在 <IMG> 标 记 中 设置 属性 USEMAP， 确 定 创建 图 像 映射 。<MAP> 标 记 的 属性 如 表 1.4 所 示 。 
表 1.4 <MAP> 标 记 的 属性 


<MAP> 标 记 的 属性 
NAME 


图 像 映射 的 名 称 
定义 图 像 映射 区 域 的 名 称 
设 定 区 域 坐标 
设 定 区 域 的 链接 地 址 
设 定 区 域 链接 的 描述 文字 


在 <MAP> 标 记 中 ， 根 据 属性 SHAPE 的 取 值 不 同 ， 相 应 坐标 的 设 定 也 不 同 。 下 面 介绍 属性 SHAPE 
的 3 种 取 值 以 及 相应 坐标 的 设 定 。 
设 定 属性 SHAPE 的 属性 值 为 rect。 属 性 SHAPE 取 值 为 rect， 表 示 和 矩形 区 域 ， 属 性 COORDS 
的 坐标 形式 为 “X1，Y1，X2，Y2”。 其 中 ，X1、Y1 代表 算 形 左上 角 的 X 坐标 和 YY 坐标， 
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X2、Y2 代表 和 矩形 右 下 角 的 义 坐 标 和 YY 坐标 。 
设 定 属性 SHAPE 的 属性 值 为 circle。 属 性 SHAPE 取 值 为 circle, 表示 圆 形 区 域 , 属性 COORDS 
的 坐标 形式 为 “X，Y，r”。 其 中 ，X、Y 为 圆心 坐标 ，r 为 圆 的 半径 。 
设 定 属性 SHAPE 的 属性 值 为 poly。 属 性 SHAPE 取 值 为 poly, 表 示 多 边 形 区 域 , 属性 COORDS 
的 坐标 形式 为 “X1，Y1，X2，Y2，…”。 其 中 ，Xn、Yn 代表 构成 多 边 形 每 一 点 的 坐标 值 ， 
n 的 取 值 为 1，2，3，…。 多 边 形 有 几 边 就 有 几 对 X、Y 坐标 。 
下 面 在 Dreamweaver 编辑 器 中 创建 图 像 映 射 ， 单 击 图 像 不 同 区 域 ， 链 接 到 不 同 的 文件 。 创 建 图 像 
映射 的 步骤 如 下 : 
(1) 在 Dreamweaver 编辑 器 中 选中 所 要 编辑 的 图 像 ， 在 属性 窗口 中 ， 有 3 种 形状 的 热点 工具 ， 分 
别 为 矩形 热点 工具 、 圆 形 热点 工具 、 多 边 形 热点 工具 。 
(2) 选择 一 种 热点 工具 ， 在 图 像 上 按 住 鼠标 左 键 拖 动 ， 确 定 所 选区 域 。 
(3) 设置 区 域 属性 ， 在 “链接 ”文本 框 中 输入 所 要 链接 书签 的 名 称 ， 在 “替代 ”文本 框 中 输入 区 
域 链接 文件 ， 如 index.php。 
(4) 重复 步骤 (2) 、(3) ， 在 图 像 上 定义 不 同 的 区 域 链接 。 


1.7.3 前 台 首 页 的 实现 过 程 


本 系统 中 所 有 的 前 台 页 面 都 采用 了 二 分 栏 结 构 ， 分 为 导航 栏 、 信 息 检 索 区 、 内 容 显示 区 和 版 权 区 4 
个 区 域 。 为 了 方便 网 站 的 日 后 维护 ， 将 这 4 个 区 域 形成 单独 的 PHP 文件 ， 然 后 应 用 include 语句 将 这 4 
个 文件 包含 进来 。 前 台 首 页 文件 的 代码 如 下 : 


例 程 01 代码 位 置 : 光盘 \TMW1\99pursey\index.php 
<table width="780" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr valign="top"> 
<td colspan="2"><?php include("top.php"):?></td> <!-- 包含 导航 文件 --> 
</t> 


<tr> 
<!-- 包含 信息 检索 文件 -> 
<td width="217" valign="top" backeround="Images/line?2.gif*><?php include("left.php"):?></td> 
<!-- 包含 内 容 显示 文件 --> 
<td width="586" valien="top" bgcolor="#FEFEF6"><?php include(“main.php"):?></td> 
</u> 
<t> 
<td colspan="2"><?php include("bottom .php"):?></td> <!-- 包含 版 权 信息 文件 --> 
</t> 


</table> 


其 中 ， 导 航 文件 top.php 页 应 用 了 图 像 映射 来 创建 文件 的 超 链 接 ， 代 码 如 下 : 


例 程 02 ”代码 位 置 : 光盘 \TM\01\99pursey\top.php 
<table width="780" height="201" border="0" align="center" cellpadding="0" cellspacing="0"> 
<t> 
<td height="38"><img src="Images/banner.gif" width="780" height="202" border="0" usemap="#Map"></td> 
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</t> 
</table> 
I 二 -图 像 映射 一 > 
<map name="Map"> 
于 人 
<area shape="rect" coords="685,8,744,27" href—="mailto:99pursey@pursey**.com"> 


es 设 为 首页 一 
<area shape="rect" coords="613,9,666,27" 
‘onClick="this.style.behavior='"url(#default#homepage)':this. as a //localhost/99pursey/index.php"):"> 
二 从 收藏 -一 = es 
<area shape="rect" coords="535.8,593,26" 
href="javascript:window.external.AddFavorite('http://localhost/99pursey/index.php',' 九 九 度 供求 信息 网 "> 


<area shape="rect" coords="448.8.516.26" href="admin/login php"> <!-- 后 台 登 录 页 --> 
<area shape="rect" coords="356.8.424.28" href="release.php"> <!-- 发 布 信息 页 --> 
<area shape="rect" coords="680.61.753.82" href="search php"> <!-- 寻 人 / 物 启示 信息 页 --> 
<area shape="rect" coords="579.62.637.82" href="sale.php"> <!-- 出 售 信息 页 --> 
<area shape="rect" coords="483,62,536,83" href="car.php"> <!-- 车 辆 信息 页 --> 
<area shape="rect" coords="385.62.441.83" href="teaching. php"> <!-- 家 教 信息 页 --> 
<area shape="rect" coords="290.61.344.83" hre 全 "seekjob.php"> <!-- 求职 信息 页 --> 
<area shape="rect" coords="678,38,740,58" href="recruitbusiness.php"> <!-- 招商 引资 信息 页 --> 
<area shape="rect" coords="579.38.635.58" href="seekbuy.php"> <!-- 求购 信息 页 --> 
<area shape="rect" coords="482,38,541,58" href="house.php"> <!-- 房屋 信息 页 --> 
<area shape="rect" coords="381,39,440,60" href="foster.php"> <!-- 培训 信息 页 --> 
<area shape="rect" coords="290.39.345.59" href="invitejob.php"> <!-- 招聘 信息 页 --> 
<area shape="rect" coords="204,39,263,86" href="index.php"> <!-- 首页 --> 

</map> 


1.8 免费 供求 信息 发 布 模块 设计 


1.8.1 免费 供求 信息 发 布 模块 概述 


免费 供求 信息 的 发 布 提供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 站 非常 重要 的 功能 ， 也 是 供求 信息 
网 站 的 核心 功能 。 

免费 供求 信息 发 布 模块 可 以 完成 11 种 不 同类 别 信息 的 发 布 。 用 户 可 以 根据 自身 需要 将 供求 信息 发 
布 到 相应 的 信息 类 别 中 〈 共 包括 11 个 信息 类 别 : 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信 
息 、 房 屋 信息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 类 型 供求 信息 ) 。 信 息 发 
布 成 功 后 ， 需 要 管理 员 进行 审核 ， 只 有 审核 成 功 的 信息 才能 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 免 费 
供求 信息 发 布 的 流程 如 图 1.60 所 示 。 


1.8.2 ”免费 供求 信息 发 布 模块 技术 分 析 


本 模块 实现 免费 供求 信息 发 布 功 能 主要 应 用 到 如 下 几 个 函数 ， 下 面 进行 详细 介绍 。 
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管理 员 


在 前 台 展 示 
发 布 的 信息 


图 1.60 免费 供求 信息 发 布 流程 图 
1.，mysql_connect() 函 数 
打开 一 个 到 MySQL 服务 器 的 连接 。 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 false。 
语法 如 下 : 
Tesource mysql_connect ( [string server [, string username [, string password [, bool new_link [, int client_flags]]]]] ) 
mysql_connect() 函 数 的 参数 说 明 如 表 1.5 所 示 。 
表 1.5 mysql_connect() 函 数 的 参数 说 阴 
参数 说 了 明 


MySQL 服务 器 。 可 以 包括 端口 号 ， 例 如 "hostname:port"， 或 者 到 本 地 套 接 字 的 路 径 ， 例 如 对 于 localhost 


ee 的 ":/path/to/socket"。 如 果 PHP 指令 mysqldefault host 未 定义 (默认 情况 ) ， 则 默认 值 是 'localhost:3306' 


username “| 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 
_password 密码 。 默 认 值 是 空 密码 


如 果 用 同样 的 参数 第 二 次 调用 mysql_connect0 函 数 , 将 不 会 建立 新 连接 , 而 将 返回 已 经 打开 的 连接 标识 。 
new_link ”| 参数 new_link 改变 此 行为 并 使 mysql_connect0 函 数 总 是 打开 新 的 连接 ， 甚 至 当 mysql_connect0 函 数 曾 
在 前 面 被 同样 的 参数 调用 过 


et client flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL CLIENT SSL、MYSQL CLIENT_COMPRESS、 
client flags 
— 8 | MYSQL CLIENT IGNORE SPACE 或 MYSQL CLIENT INTERACTIVE 
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2. mysql_select_db() 函 数 

选择 MySQL 数据 库 。 如 果 成 功 返 回 tue， 失 败 返回 false。 语 法 如 下 : 

bool mssql select db ( string database_ name [. resource link identifier] ) 

Imysql_select_db0 函 数 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没有 指 


定 连接 标识 符 , 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ,本 函数 将 无 参数 调用 mysql_connectO 
函数 来 尝试 打开 一 个 并 使 用 之 。 每 个 其 后 的 mysql_query0 函 数 调用 都 会 作用 于 活动 数据 库 。 


下 面 应 用 mysql_connect() 函 数 连接 MySQL 服务 器 ， 然 后 应 用 mysql select_db0 函 数 连接 MySQL 


数据 库 ， 代 码 如 下 : 
<?php 
Sconn = mysql_connect("localhost", "root", "root"): // 连 接 MySQL 服 务 器 
$db=mysql_select_db("db_pursey", $conn) or die ("数据 库 连 接 失败 :" .mysql error0): /连接 数据 库 db_pursey 
mysql_query("set names gb2312"): /采用 GB2312 编 码 方式 


?> 

3. mysql_query() 函 数 

mysql_queryO 函 数 用 来 根据 连接 标识 符 向 该 数据 库 服务 器 的 当前 数据 库 发 送 查 询 。 语 法 如 下 ; 
int mysql query(string query ,int [link identifier]); 

其 中 ，query 是 查询 字符 串 ，link_identifier 是 数据 库 连 接 标 识 符 。 

mysql_query0 函 数 在 执行 成 功 时 返回 一 个 结果 标识 符 ， 失 败 时 返回 false。 

4. date() 函 数 

date(O) 函 数 主 要 用 于 格式 化 一 个 本 地 时 间 / 日 期 。 语 法 如 下 : 

string date ( string format , int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 ,其 中 参数 timestamp 是 可 选 的 ， 


默认 值 为 tme0， 即 如 果 没 有 给 出 时 间 戳 ， 则 使 用 本 地 当前 时 间 。 


date(O) 函 数 的 参数 format 的 格式 化 选项 如 表 1.6 所 示 。 


表 1.6 参数 format 的 格式 化 选项 
说 了 明 


小 写 的 上 午 和 下 午 值 ， 返 回 值 am 或 pm 


大 写 的 上 午 和 下 午 值 ， 返 回 值 AM 或 PM 
Swatch Internet 标准 时 ， 返 回 值 000 一 999 


月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 01 一 31 
星期 中 的 第 几 天 ,文本 格式 ，3 个 字母 ， 返 回 值 Mon 一 Sun 


月 份 ， 完 整 的 文本 格式 ， 返 回 值 January 一 December 


小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 1 一 12 


小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 0 一 23 


小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 01~~12 
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小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 00 一 23 
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续 表 
参数 说 明 
有 前 导 零 的 分 钟 数 ， 返 回 值 00 一 59 
判断 是 否 为 夏令 时 ， 返 回 值 如 果 是 夏令 时 为 1， 否 则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 1 一 31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 Sunday 一 Saturday 
判断 是 否 为 头 年 ， 返 回 值 如 果 是 闲 年 为 1， 否 则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 01 一 12 
3 个 字母 缩写 表示 的 月 份 ， 返 回 值 Jin 一 Dec 
数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 1 一 12 
与 格林 威 治 时 间 相 差 的 小 时 数 ， 例 如 : 0200 
RFC 822 格式 的 日 期 ， 例 如 : Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 00 一 59 
每 月 天 数 后 面 的 英文 后 缀 ，2 个 字符 ， 例 如 st，nd，rd 或 者 也 。 可 以 和 j 一 起 使 用 
指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
星期 中 的 第 几 天 ， 数 字 表 示 ， 返 回 值 为 0 一 6 
ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
它 年 份 中 的 第 几 天 ， 返 回 值 0 一 366 
时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 
-43200 一 43200 


全 注意 : 有 效 的 时 间 玲 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 日 
03:14:07 ( 此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 ) 。 在 Windows 系统 中 此 范围 限 
制 为 从 1970 年 1 月 1 日 到 2038 年 1 月 19 日 。 
例如 应 用 date0 函 数 格式 化 一 个 日 期 ， 并 输出 日 期 的 值 ， 代 码 如 下 : 


<?php 
echo date("y:m:d"): 
?> 


<|zlz cll- lv le | |o ls lz le Ic | | | 


结果 为 : 07:07:11 
1.8.3 ”免费 供求 信息 发 布 模块 的 实现 过 程 
国 ”免费 供求 信息 发 布 使 用 的 数据 表 : tb_info 


用 户 通 过 单 击 前 台 页 面 导航 栏 的 “我 要 发 布 ” 超 链 接 ， 进 入 信息 发 布 页 面 ， 如 图 1.61 所 示 。 程序 
会 先 验证 用 户 是 否 输入 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 
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数据 


库 中 插入 记录 ， 完 成 发 布 操作 。 


必 习 * 必 地 产 格 在 08 征 5 月 忆 绕 工 1 
有 酒店 正式 成 立 ? 欢 地 您 的 至 7 
“名 所 "六 务 印刷 地 作 允 加盟 ， 整 消 ， 通 风 好 ， 有 用 千 简 ， 有 电视 ， 提 供 扫 


“" 诚 部 电子 商务 吉林 省 战 喇 合作 羽 伴 亲情 化 专人 管理 ， 是 理 起 的 公 窗 选择 。 
“ 九 九 度 洪 求 信息 在 诡 开 通 9 所 有 全 


周 芒 息 块 速 检索 


图 1.61 免费 供求 信息 发 布 网 页 
在 功能 导航 页 top.php 中 ， 在 “我 要 发 布 ”图 像 上 添加 图 像 映射 ， 并 绘制 热 区 ， 代 码 如 下 : 
例 程 03 ”代码 位 置 : 光盘 \TM\01\99pursey\top.php 


<map name="Map"> // 图 像 映 射 
<area shape="rect" coords="356.8,424,28" href="release.php"> 
</map> 


在 信息 发 布 页 面 选 择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添 加 空 信息 ， 


在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 
素 是 否 为 空 值 。 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 代 码 如 下 : 


例 程 04 ”代码 位 置 : 光盘 \TMW1\99purseywelease_contentphp 
<script language="javascript"> 


function checkform(form) { // 自 定义 一 个 JavaScript 函 数 checkform0 
for(i=0:i<form.length:it+){ /应 用 for 循 环 语句 检索 form 表 单元 素 的 值 是 否 为 空 
if(form.elements[i].value—""){ // 如 果 form 表 单 中 某 个 元 素 值 为 空 
alert(" 请 将 发 布 信息 填写 完整 ! "); /弹出 提示 信息 
form.elements[i] .focus(): // 将 焦点 的 值 定位 到 为 空 值 的 表单 元 素 
return false: // 返 回 焦点 
} 
外 
1 
</script> 


创建 与 数据 库 db_pursey 的 连接 ， 代 码 如 下 : 
例 程 05 ”代码 位 置 光盘 \TM\01\99pursey\conn\conn.php 


<2php 
Slink=mysql_connect("localhost","root"."root"): // 连 接 MySQL 服 务 器 
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© mysql select db("db pursey "$link): /连接 MySQI 数 据 库 文件 dtb_pursey 
© mysql query("setnames gb2312"): // 采 用 GB2312 编 码 方式 

2 
< 代码 贴 二 


@ mysql connect0 函 数 : 连接 MySQL 服务 器 ， 详 细 讲 解 参 见 1.8.2 节 。 
四 mysql_ select db0 函 数 : 连接 MySQL 数据 库 文件 ， 详 细 讲 解 参 见 1.8.2 节 。 
图 mysql_ query0 函 数 : 该 函数 用 来 向 数据 库 服务 器 发 送 编码 方式 ， 详 细 讲 解 参 见 1.8.2 节 。 


提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信 息 
添加 成 功 ， 则 弹出 成 功 的 信息 提示 ， 否 则 弹出 失败 的 提示 信息 。 代 码 如 下 : 
例 程 06 ”代码 位 置 : 光盘 \TMW1\99purseyelease_okphp 


<2php 
include("conn/conn.php"): // 连 接 数 据 库 文件 
Stype=$_POST[typel: // 获 取信 息 类 型 
Stitle=$_ POST[title]: /获取 信息 标题 
Scontent=$_POST[content]: // 获 取信 息 内 容 
Slinkman=$_POST[linkman]:; // 获 取 联 系 人 
Stel=$_POST[tel]: // 获 取 联系 电话 
@ Sedate=date("Y-m-dh:is"): // 获 取 发 布 时 间 
@ $sql=mysql query("insert into tb_info(type.title,.content,linkman.tel,.checkstate,edate) 
values('$type'.'$title','$content','$linkman','$tel'.0,'$edate’)"); // 将 免费 的 供求 信息 添加 到 数据 表 中 
这 $sqD){ // 如 果 添加 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert( 恭 喜 您 ， 信 息 发 布 成 功 ! ');window.location.href='release.php';</script>"; 
jelse{ /如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert( 对 不 起 ， 信 息 发 布 失败 ! ):history.back0:</script>"; 
} 
i 
Ah 代码 贴 十 


@ date: 格式 化 一 个 本 地 时 间 / 日 期 ， 详 细 讲解 参见 1.8.2 节 。 
@ insert into: 向 指定 的 数据 表 中 添加 数据 信息 。 


1.9 信息 检索 模块 设计 


1.9.1 “信息 检索 模块 概述 


信息 检索 是 对 已 存在 于 数据 库 中 的 数据 按 条 件 进 行 筛选 浏览 ， 是 查看 历史 信息 和 确认 数据 操作 最 
为 快速 、 有 效 的 办 法 。 信 息 检索 模块 主要 通过 选择 信息 类 型 和 输出 查询 关键 字模 糊 查 询 供求 信息 资源 ， 
并 输出 查询 结果 。 考 虑 到 供求 信息 的 信息 量 较 大 ， 因 此 本 模块 对 与 查询 关键 字 相 匹 配 的 查询 结果 进行 
描 红 ， 从 而 方便 用 户 的 浏览 。 信 息 检索 模块 的 示意 图 如 图 1.62 所 示 。 
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设置 查询 条 件 


1.62 ”信息 检索 模块 的 示意 图 


1.9.2 ”信息 检索 模块 技术 分 析 


在 对 数据 进行 查询 后 ， 最 终 需要 将 查询 结果 显示 在 页 面 中 反馈 给 浏览 者 。 在 PHP 中 ， 查 询 结果 
的 显示 方式 有 很 多 种 ， 最 常用 的 就 是 表格 显示 方式 。 因 为 采用 这 种 方式 显示 的 数据 条 理 清 晰 、 简 捷 
明了 。 

在 利用 表格 显示 查询 结果 时 , 通常 是 将 查询 结果 保存 在 结果 集中 , 然后 需要 使 用 do…while 循环 将 
其 查询 结果 显示 出 来 。 需 要 注意 的 是 ， 需 要 先 判断 查询 结果 是 否 为 室 ， 只 有 查询 结果 不 为 空 时 ， 才 可 
以 使 用 循环 语句 显示 数据 。 为 了 使 读者 更 好 地 理解 通过 表格 显示 查询 结果 ， 下 面 给 出 其 实现 流程 图 ， 
如 图 1.63 所 示 。 

考虑 到 用 户 不 可 能 全 面 了 解数 据 表 中 的 数据 信息 ， 例 如 不 能 确定 所 要 查询 信息 的 内 容 、 查 询 的 主 
题 等 ， 这 时 就 需要 使 用 like 进行 模糊 查询 。like 关键 字 需 要 使 用 通配符 在 字符 串 内 查找 指定 的 模式 , 所 
以 读者 需要 了 解 通配符 及 其 含义 。 通 配 符 的 含义 如 表 1.7 所 示 。 
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图 1.63 信息 检索 模块 流程 图 
表 1.7 like 关键 字 中 的 通配符 及 说 明 


由 零 个 或 更 多 字符 组 成 的 任意 字符 串 


用 于 指定 范围 ， 例 如 [A~F]J， 表 示 A~F 范围 内 的 任何 单个 字符 
表示 指定 范围 之 外 的 ， 例 如 [^A~F] 表 示 A~F 范围 以 外 的 任何 单个 字符 


如 果 想 查询 包含 “女子 公寓 ”的 信息 ， 可 以 使 用 like 运算 符 配 合 通 配 符 “%” 完 成 。 其 SQL 语句 
如 下 : 

select * 位 om tb_info where content like '% 女 子 公寓 %'; 

如 果 想 查找 信息 类 型 为 “公寓 信息 ”或 者 内 容 为 “女子 公寓 ”的 信息 时 配合 or 运算 符 来 使 用 。 其 
SQL 语句 如 下 : 

select* from tb_info 。 where type=' 公 寓 信 息 ' or content like'% 女 子 公 寓 % 

本 模块 实现 付费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 : 


$type=$_POST[typel: // 获 取信 息 类 型 

Scontent=$_POST[content]: // 获 取 查询 关键 字 

$sql1=mysql query("select * from tb leaguerinfo where checkstate=1 and type= Stype' and content like'9%6$content96 or 
title like'%$content%' or linkman like'9o$contento%6' or tel like'%6Scontent90”): 

Sinfol=mysql fetch array($sql]): /检索 付费 的 信息 


本 模块 实现 免费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 : 


$sql=mysql_query("select * fiom tb info where checkstate=1 and type='$type' and content like'%$content%' or title 


她 $ 
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like'%S$content%%' or linkman like'%S$content%6' or tel like'%S$content%"): 
S$info=mysql fetch array($sq]); 1/ 检索 免费 的 信息 


名 注意 : 当 满足 数据 表 中 多 个 字段 中 的 任 一 字段 时 ， 可 以 使 用 or 运算 符 将 多 个 条 件 连接 。 


另外 ， 由 于 搜索 的 内 容 中 文字 比较 多 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信息 ， 所 以 在 搜索 
引擎 中 加 入 了 描 红 功 能 。 描 红 功 能 主要 用 str_ireplace0 函 数 实现 ,该 函数 的 具体 讲解 可 参见 本 章 的 1.14.1 
节 。 


1.9.3 ”信息 检索 模块 的 实现 过 程 


国 ”信息 检索 模块 使 用 的 数据 表 : tb_info、tb_leaguerinfo 

在 开发 信息 检索 模块 时 ， 由 于 该 网 站 含有 大 量 的 数据 信息 ， 为 了 方便 用 户 浏览 网 站 信息 ， 需 要 添 
加 复合 条 件 查 询 实现 搜索 功能 。 在 信息 检索 区 的 “关键 字 ” 文 本 框 中 输入 欲 查询 的 关键 字 ， 在 “条 件 ” 
下 拉 列 表 框 中 选择 要 搜索 的 信息 类 型 ， 然 后 单 击 “ 开 始 搜 索 ” 按 钮 ， 对 指定 条 件 的 记录 进行 检索 并 输 
出 结果 集 到 浏览 器 ， 同 时 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信 息 ， 本 模块 对 查询 关键 字 进 行 描 
红 。 运 行 结果 如 图 1.64 所 示 。 


本 -LETLETLTRIRTTT 本 ETIEIZITR 


FHI EE 
WW nc110.cOM, 全 人 


联系 人 : 个 光 生 。 联系。 
只 识 汪 息 」 白领 关子 公 宙 ”200T-12-20 09-33 号 
二 要 再 特 近 ,2 税 ， 提 人 ?人间 ，8 人间。 对 ， 宣 内 于， 吾 壬 ,中风 好 ,有 用 生生， 有 电 
祖 ， 保志 水 。 作 属 香 条 生 涉世 人 答 理 ， 是 要 的 公 寅 由 控 ， 
联系 人 : 王 阿 访 联系 电 活 : bgegey 
0 浴 训导 各】 必 怕 "文子 公 富 “200T-12-0 09:40 名 


可 从 宣 位 于 和 用 绝 . 豆 便 大 人 ， 环 二 好 ， 现 入 往 人 员工 作 入 定 素质 宫 欢迎 有 各 定 工作 壬 二 昼 妇 芒 
入 在， 有 不由 虽 环 训 护 ， 


图 1.64 信息 检索 页 面 的 运行 结果 
信息 检索 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 1.8 所 示 。 
表 1.8 信息 检索 页 面 所 涉及 的 重要 表单 元 素 


名 称 


forml 


重要 属性 
method="post" action="findinfo.php" 


id="content" size="20" 


表单 


Content 
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名 称 重要 属性 
<select name="type"> 
<option value=" 招 聘 信 息 ">- 招 聘 信息 -</option> 


type <option value=" 求 职 信息 ”selected>- 求 职 信 息 
-</option> 
</select> 


search imag g onClick="return chkinput(form)" 


应 用 JavaScript 脚本 自 定义 一 个 chkinput0 函 数 ， 实 现 对 表单 提交 的 信息 进行 验证 ， 代 码 如 下 : 


例 程 07 代码 位 置 ， 光盘 \TM\01\99pursey\left.php 
<script language="javascript"> 


function chkinput(form) { // 自 定义 一 个 chkinput0 函 数 
if(form.content.value—""){ /判断 如 果 查 询 关键 字 文 本 框 等 于 空 
alert(" 请 输入 查询 关键 字 !"): // 则 弹出 提示 信息 
form.content.select|:; // 重 新 定位 焦点 
return false: // 返 回 表单 元 素 
} 
} 
</script> 


将 表单 信息 提交 到 数据 处 理 页 ， 连 接 数 据 库 文件 ， 接 收 表单 信息 ， 然 后 用 mysql_query0 函 数 向 服 
务 器 发 送 SQL 语句 ， 检 索 与 查询 关键 字 相 匹配 的 信息 资源 。 代 码 如 下 : 
例 程 08 ”代码 位 置 : 光盘 \TMVWO1\99pursey\findinfo php 


<?php 

include("conn/conn.php"): // 连 接 数 据 库 文件 
$type=$_POST[type]: // 获 取信 息 类 型 
Scontent=$_POST[content]: // 获 取 查 询 关 键 字 


$sqll=mysql_query("select * from tb_leaguerinfo where checkstate=] and type='Stype' and content like'%$content%' or 
title like'%$content%o' or linkman like'%$content%%' or tel like'%$content%6"): 

Sinfol=mysql fetch_array($sql1): /检索 付费 的 供求 信息 

$sql=mysql query("select * firom tb info where checkstate=1 and type='$type' and content like'%S$content%o' or title 
like'%oScontent%o' or linkman like'%oScontent9%o' or tel like'%S$content%0"): 

Sinfo=mysql_fetch_array($sql): /检索 免费 的 供求 信息 

?> 


[DD 说 明 : 信息 检索 需要 从 免费 信息 表 tb info 和 付费 信息 表 tb leaguerinfo 中 获取 数据 ， 因 此 需要 向 
MySQL 服务 器 传递 两 条 SQL 语句 。 
用 do…while 循环 语句 输出 付费 信息 与 查询 关键 字 相 匹配 的 信息 资源 ， 并 用 str_ireplace0 函 数 对 查 
询 关 键 字 实现 描 红 功 能 ， 代 码 如 下 : 
例 程 09 ”代码 位 置 : 光盘 \TM\01\99pursey\findinfo.php 
<!- 下 面 输出 的 是 付费 信息 与 查询 关键 字 相 匹配 的 信息 -> 


<?php 
if($infol){ // 如 果 检 索 到 了 付费 信息 
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dof // 则 用 do…while 循 环 语句 输出 付费 信息 
化 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<t> 


<td height="26"> 
<!- 应 用 str_ireplace0 函 数 对 查询 关键 字 进行 描 红 -> 
<!-- 对 与 查询 关键 字 所 匹配 的 信息 类 型 进行 描 红 -> 
[<?php echo str_ireplace($content."<font color=#FF0000'>".$content."</font>",$infol[type]):?>)] nbsp: 
<!-- 对 与 查询 关键 字 所 匹配 的 信息 标题 进行 描 红 -> 
<?php echo str_ireplace($content,"<font color=#FF0000>".$content."</font>",$info1[title]):?>&nbsp:é&nbsp:; 
<!-- 对 与 查询 关键 字 所 匹配 的 发 布 时 间 进 行 描 红 -> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$infol[edate]):?></td> 
</t> 
<t> 
<td height="26">&nbsp:&nbsp:&nbsp:&nbsp: 
<!-- 对 与 查询 关键 字 所 匹配 的 信息 内 容 进行 描 红 -> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$infol[content]):?></td> 
</t> 
<t> 
<td height="26">&nbsp: 联 系 人 : 
<!-- 对 与 查询 关键 字 所 匹配 的 联系 人 进行 描 红 -> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$infol[linkman]);?>&nbsp; 
<!-- 对 与 查询 关键 字 所 匹配 的 联系 电话 进行 描 红 -> 
联系 电话 : <?php echo str_ireplace($content."<font color=#FF0000'>".$content."</font>",$info1[tel]):?> 
es 一 > 
</ltd> 
</u> 
</table> 
<?php 
}while($infol=mysql fetch_array($sql1)): // 循 环 语句 结束 
?> 


[说 明 : 免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 代 码 部 分 略 ， 详 见 本 书 附 赠 光 盘 ， 
免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 下 面 给 出 实现 过 程 的 核心 代码 结构 。 
例 程 10 ”代码 位 置 : 光盘 \TMW1\99pursey\findinfo php 


< 一 一 一 一 一 一 下 面 输出 的 是 免费 信息 与 查询 关键 字 相 匹配 的 信息 : 一 一 一 一 一 
<?php 
ifSinfo){ // 如 果 检索 到 了 免费 信息 
dof // 则 用 do…while 循 环 语句 输出 付费 信息 
?> 
oo // 免 费 信息 的 输出 方式 与 付费 信息 的 类 似 , 代码 略 
<?php 
} while($info=mysql_fetch_array($sql)): //do…while 循 环 语句 结束 
// 站 条 件 语句 结束 


如 果 在 免费 信息 表 和 付费 信息 表 中 没有 检索 到 与 查询 关键 字 相 匹配 的 数据 ， 则 弹出 提示 信息 ， 代 
码 如 下 : 
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例 程 11 代码 位 置 : 光盘 \TM\01\99pursey\findinfo.php 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td align="center"> 您 检索 的 信息 资源 不 存在 ! </td> 
</t> 
</table> 


1.10 后 台 首 页 设计 


1.10.1 后 台 首 页 概述 


程序 开发 人 员 在 设计 网 站 后 台 主 页 时 ， 主 要 从 后 台 管 理 人 员 对 功能 的 易 操 作 实用 性 、 网 站 的 易 维 
护 性 考虑 ， 因 此 采用 了 框架 技术 。 九 九 度 供求 信息 网 后 台 首 页 主要 包含 以 下 内 容 : 

回 ”发布 付费 的 供求 信息 (包括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信 息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 ) ， 以 及 付费 信息 的 浏览 、 
审核 及 删除 功能 。 
免费 信息 的 浏览 、 审 核 及 删除 功能 。 
企业 广告 信息 的 发 布 、 浏 览 、 前 台 推荐 显示 、 删 除 功能 。 

网 站 首页 : 为 管理 员 进 入 前 台 提 供 一 个 入 口 。 
退出 登录 : 注销 当前 用 户 。 

下 面 看 一 下 本 案例 中 提供 的 后 台 首页 ， 如 图 1.65 所 示 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TMN\O1\ 
99pursey\admin\index.php。 
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图 1.65 ”供求 信息 网 站 后 台 首页 
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1.10.2 ”后台 首页 技术 分 析 


九 九 度 供求 信息 网 后 台 采 用 框架 技术 进行 页 面 布局 。 所 谓 框架 就 是 网 页 的 各 部 分 为 相互 独立 的 网 
页 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 ， 重 复出 现 的 内 
容 被 固定 下 来 ,每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 

使 用 框架 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 在 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 。 首 先 在 
Dreamweaver 8 中 创建 一 个 “ 左 一 中 一 右 ” 的 框架 集 ， 然 后 在 标识 四 中 添加 “上 一 中 一 下 ”的 框架 集 ， 
最 后 在 标识 @ 中 添加 一 个 “ 左 一 右 ” 的 框架 集 ， 从 而 完成 一 个 完整 的 后 台 框 架 。 构 建 框架 的 流程 如 


图 1.66 所 示 。 


a E> 
E33 Ea 
S. S. 
-< -< 
FE Ei 
号 号 
性 


图 1.66 网 站 后 台 框架 流程 
使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 下 面 详细 介绍 框架 的 基本 结构 、 设 置 框架 集 属性 和 框架 


属性 。 

1. 框架 网 页 的 基本 结构 

框架 网 页 通过 一 个 或 多 个 FRAMESET 和 FRAME 标记 来 定义 。 在 框架 网 页 中 ,将 FRAMESET 标 
记 置 于 HEAD 标记 之 后 ， 以 取代 BODY 标记 的 位 置 ， 当 客户 端 浏览 器 不 支持 框架 网 页 时 ,还 可 以 使 用 
NOFRAMES 标记 给 出 框架 不 能 被 显示 时 的 替换 内 容 。 框 架 网 页 的 基本 结构 可 以 表示 如 下 : 


<html> 
<head> 
<title> 基 本 框架 页 </title> 
</head> 
<frameset> 

<frame> 
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<frame> 
</frameset> 
<noframes> 
<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 页 面 的 显示 ! 
</body> 
</noframes> 
</html> 


2. 设置 框架 集 的 属性 
框架 集 包 含 如 何 组 织 各 个 框架 的 信息 , 可 以 通过 FRAMESET 标记 来 定义 。 框 架 是 按照 行 和 列 来 组 
织 的， 可 以 使 用 FRAMESET 标记 的 下 列 属性 对 框架 的 结构 进行 设置 。 
(1) 左右 分 割 窗口 属性 COLS 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 左右 分 割 窗口 属性 COLS 实现 。 其 语法 
格式 如 下 : 
<frameset cols="value,value,..."> 
<frame> 


<frame> 
</frameset> 


属性 说 明 如 下 : 
value: 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 列 ， 其 中 第 1 列 占 浏览 器 窗口 宽度 的 20%， 第 2 列 为 
120 像素 ， 第 3 列 为 浏览 器 窗口 的 剩余 部 分 的 框架 ， 代 码 如 下 : 
<frameset cols="20%,120,*" > 
<frame> 


<frame> 
</frameset> 


避 技巧 : 如 果 将 COLS 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ; 如 果 将 COLS 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 116， 中 间 的 框架 占 窗 口 宽度 
的 1/3， 右 边 的 框架 占 窗口 宽度 的 1/2。 


(2) 上 下 分 割 窗 口 属性 ROWS 
在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 上 下 分 割 窗口 属性 ROWS 实现 。 其 语法 
格式 如 下 : 
<frameset rows="value,value,......"> 
<frame> 


<frame> 
</frameset> 


属性 说 明 如 下 : 
value: 用 于 指定 各 个 框架 的 行 高 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 ， 设 置 方 
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法 与 COLS 属性 类 似 。 例 如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 行 ， 其 中 的 第 1 行 占 浏览 器 窗口 宽 
度 的 20%， 第 2 行为 120 像素 ， 第 3 行为 浏览 器 窗口 的 剩余 部 分 的 框架 。 其 代码 如 下 : 
<frameset rows="20%,120,*" > 
<frame> 


<frame> 
</frameset> 


(3) 框架 边框 显示 属性 FRAMEBORDER 

该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 默 认 值 ) 或 0 (不 显示 边框 )。 

(4) FRAMESPACING 

该 属性 用 于 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 如 果 不 设置 该 属性 ， 则 框架 之 间 没 有 间隔 。 

(5) 指定 边框 宽度 属性 BORDER 

该 属性 用 于 指定 边框 的 宽度 ， 只 有 FRAMEBORDER 属性 为 1 时 有 效 。 

3. 设置 框架 的 属性 

使 用 FRAME 标记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 框 架 是 否 包含 滚动 条 以 及 在 框架 中 显 
示 的 网 页 等 。 其 语法 格式 如 下 : 

<frame name='" 框 架 名 称 " src=" 文 件 " frameborder=" 数 值 " scrolling=" 值 " [noresize] > 

属性 说 明 如 下 : 
name: 指定 框架 的 名 称 。 
sIc: 指定 在 框架 中 显示 的 网 页 文件 (包括 HIML、ASP 等 网 页 文件 ) 。 
frameborder: 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 ) 或 0 (不 显示 ) 。 默 认 值 为 1。 
scrolling: 指定 框架 是 否 包含 滚动 条 。 如 果 将 该 属性 设置 为 yes， 则 框架 包含 滚动 条 ; 若 将 该 
属性 设置 为 no， 则 框架 不 包含 滚动 条 ;如 果 将 该 属性 设置 为 aato， 则 在 需要 时 包含 滚动 条 。 
noresize: 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 。 


1.10.3 ”后 台 首页 的 实现 过 程 


多 办 凶 


加 


根据 1.10.1 节 和 1.10.2 节 的 页 面 概述 及 实现 技术 分 析 ， 需 要 分 别 创建 实现 各 区 域 的 PHP 文件 ， 如 
实现 Banner 广告 栏 的 top.php、 功 能 导航 栏 的 leftphp、 内 容 显示 区 的 main.php 和 页 尾 文件 bottom.php 
等 。 下 面 给 出 实现 该 系统 后 台 框 架 布局 的 完整 代码 。 


例 程 12 ”代码 位 置 : 光盘 VTMVW1\99purseyvadminindex.php 
<frameset rows="*" cols="1*.1005.1*" framespacing="0" frameborder="NO" border="0"> 
<frame src="blank.php" name="left" scrolling="NO" noresize> <!-- 设 置 空 框架 页 --> 
<frameset rows="1005.*" cols="*" framespacing="0" frameborder="NO" border="0"> 
<frameset rows="94.*.190" cols="*" framespacine="0" fr ameborder="NO" border="0"> 
<frame src="top.php" name="topFrame" scrolling="NO" noresize> 
<frameset rows="*" cols="229,*" framespacine="0" frameborder="NO" border="0"> 
<frame src="left.php" name="leftFrame" scrolling="NO" noresize> 
<frame src="main.php" name="mainFrame" scrolling="NO" noresize> 
</frameset> 
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<frame src="bottom.php" name="bottomFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php" name="right" scrolline="NO" noresize> 
</frameset> 
<frame src="blank.php"></frameset> <!-- 设 置 空 框架 页 --> 
<noframes><body> 
</body></noframes> 
各 注意 : 在 建设 Web 网 站 时 ， 如 何 让 不 同 分 辨 率 的 用 户 都 能 看 到 网 页 的 最 佳 效 果 是 程序 员 在 设计 之 
初 所 要 考虑 的 首要 问题 。 为 了 使 屏蔽 的 分 辨 率 在 大 于 1024X768 像素 的 设置 时 仍然 处 于 居 
中 显示 , 只 需要 在 设置 框架 布局 时 ,在 主 框架 两 侧 各 设置 一 个 宽度 相同 的 blank.php 空 页 即 可 。 


1.11 付费 供求 信息 发 布 模 块 设计 


1.11.1 付费 供求 信息 发 布 模块 概述 


付费 供求 信息 的 发 布 提供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 站 非常 重要 的 功能 ， 也 是 供求 信息 
网 站 的 盈利 点 。 企 业 或 用 户 可 以 根据 自身 需要 对 供求 信息 先进 行 付费 ， 付 费 后 由 管理 员 在 后 台 将 供求 
信息 发 布 到 相应 的 信息 类 别 中 〈 共 包括 11 个 信息 类 别 : 招聘 信息 、 求 职 信息 、 培 训 信息 、 公 寓 信息 、 
家 教 信息 、 车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 兑 出 况 、 寻 求 合作 、 企 业 广 告 等 类 型 供求 信息 ) 。 供 
求 信 息 成 功 发 布 后 ， 管 理 员 需要 在 后 台 对 发 布 的 信息 进行 审核 ， 如 果 审核 通过 后 ， 则 显示 在 前 台 相 应 
的 信息 类 别 网 页 中 。 付 费 供求 信息 发 布 流程 图 如 图 1.67 所 示 。 


在 前 台 展 示 
发 布 的 信息 
图 1.67 付费 供求 信息 发 布 流程 图 
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1.11.2 ”付费 供求 信息 发 布 模块 技术 分 析 


付费 供求 信息 与 免费 供求 信息 不 同 的 是 ， 付 费 供求 信息 不 仅 需要 收取 一 定 的 费用 ， 而 且 还 需要 一 
定 的 时 间 限 制 。 例 如 ， 网 站 要 求 一 个 月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 20 
元 ， 那 么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进 行 显示 时 ， 不 需要 管理 员 进行 手动 管理 ， 而 是 通过 程序 
直接 计算 出 信息 显示 的 截止 时 间 。 

信息 显示 的 截止 时 间 =“ 系 统 当前 日 期 ”+ “信息 的 有 效 天 数 〈 与 用 户 交纳 的 信息 费 相 关 〉”。 

自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 


$days=$_ POST[days]: // 通 过 表 单传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)): // 信 息 显示 的 截止 时 间 
[0D 说 明 : 信息 的 有 效 天 数 与 用 户 交纳 的 信息 费 相 关 ， 交 费 不 通过 本 程序 完成 因此， 信息 的 有 效 天 
数 需要 管理 员 手动 添加 。 


1.11.3 ”付费 供求 信息 发 布 模块 的 实现 过 程 


国 ”付费 供求 信息 发 布 使 用 的 数据 表 : tb_leaguerinfo 

用 户 通过 单 击 页 面 导航 区 的 “付费 信息 ” 超 链接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 1.68 所 示 。 填 写 
真实 有 效 的 付费 信息 ， 单 击 “ 发 布 信 息 ” 按 钮 ， 程 序 会 先 验 证 用 户 输 入 的 信息 ， 若 验证 失败 ， 则 返回 
信息 发 布 页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 


全 


四 您 现下 的 位 置 : 九 九 度 洪 求 入 是 同 》 后 台 营 理 系统 


图 做 硬 信息 显示 方式 


一 
大 审改 关 者 一 一 一 一 一 一 | 
六 人 示 市 什 个 全 部 


—| 信息 类 别 ; 家教 信息- ”到 反 是 否 付 费 
| DD .5 71. | 
全 付费 信息 显示 方式 得 二 相机 
妇 人 Ht | 
CH 得 全 术 付 要 个 全 孝 
TE 
则 全 业 广告 显示 方式 


大 扒 基 关 老 - 
个 推荐 全 未 推举 个 全 部 


本 
= 
2 


红 本 站 0 于 五 ， 于 全 
外 请 ， 有 三 年 以 上 幼教 经 验 者 优 灶 ， 介 名 


畴 席 供求 信息 同 www_bei10. on 版权 新 有 


图 1.68 付费 供求 信息 发 布 页 面 运行 结果 
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在 左 侧 框架 leftphp 页 面 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 ， 代 码 如 下 : 


例 程 13 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\left.php 
<form name="form1" method="post" action="release content.php" target="mainFrame"> 
<input name="imageField" type="image" class="inputl" src="images/btn fufei.gif’ width="210" height="39" 


border="0"> 


</form> 


单 击 “ 付 费 信息 ”按钮 ， 将 信息 页 release_content.php 中 的 内 容 显 示 在 框架 显示 页 mainFrame 中 。 
付费 供求 信息 发 布 页 面 主要 用 于 发 布 付费 的 供求 信息 , 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 1.9 


所 示 。 
表 1.9 ”付费 供求 信息 页 面 所 涉及 的 重要 表单 元 素 

名 称 | 元 素 类 型 重要 属性 含义 
forml form method="post" action="release_ok.php" 表单 

<select name="type”"> 

<option value=" 招 聘 信息 ">- 招 聘 信 息 -</option> 

<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 
type select a . 信息 类 型 

<option value=" 寻 人 / 物 启 示 ">- 寻 人 / 物 启示 -</option> 

</select> 

flag checkbox class="inputl” value="1" checked “是 否 付费 ” 复 选 框 
title text size="50" 信息 标题 
content textarea cols="55" rows="8" 信息 内 容 
linkman text size="30" 联系 人 
tel text size="30" 联系 电话 
days text 有 效 天 数 
imageField image src="images/fa.jpe" onClick="retum checkform(form):" “发 布 信息 ”按钮 


在 付费 信息 发 布 页 面 选 择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添 加 空 
信息 ， 在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定 义 一 个 checkform0 函 数 ， 验 证 提交 的 表 
单 各 元 素 是 否 为 空 值 ， 如 果 为 室 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。checkfomm() 
函数 的 代码 部 分 与 例 程 04 相同 ， 这 里 不 再 著述 。 

提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信 息 
添加 成 功 ， 则 弹出 成 功 的 信息 提示 ， 否 则 弹出 失败 的 提示 信息 。 代 码 如 下 : 


例 程 14 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\release_ok.php 
<2php 

include("../conn/conn.php"): 

$type=$_POST[typel: 

$flag-$ POST[flag]: 

Stitle=$_POST[title]: 

$content=$_ POST[content]: 

$linkman=$ POST[linkman]: 


/连接 数据 库 文件 
/获取 信息 类 型 
/获取 付款 状态 
/获取 信息 标题 
/获取 信息 内 容 
/获取 联系 人 
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$days=$_ POST[days]: /获取 发 布 时 间 
Stel-$_POST[tel]: /获取 联系 电话 
Ssdate=date("Y-m-d"): // 当 前 系统 时 间 

@ sshowday=date("Y-m-d",(timeQ+3600*24*Sdays)): // 获 取信 息 的 有 效 时 间 


$sql=mysql query("insert into tb_leaguerinfo(type,title.content.linkman.tel.sdate.showday.checkstate) 
values('$type'.'$title','$content','$linkman','$tel','$sdate'.'$showday’.$flag)"): // 将 付费 的 供求 信息 添加 到 数据 表 中 
if($sqD{ /如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
© echo "<script>alert(' 信 息 发 布 成 功 ! "): parent.mainFrame.location.href='release_content.php';</script>"; 
Jelse{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 信 息 发 布 失败 ! "):history.backO:</script>": 
3 


?> 
外 代码 贴 二 
@ date("Y-m-d".(time0+3600*24*Sdays)): 表示 信息 的 有 效 时 间 = 当 前 期 日 期 + 付费 期 限 . 应 用 time0) 函 数 获取 当前 日 期 


时 间 戳 ， 付 费 期 限 的 时 间 戳 等 于 3600 秒 x 24 小 时 x 指定 天 数 ， 并 通过 date0) 函 数 格式 化 为 指定 日 期 格式 。 
@ parentmainFrame ,location hre 人 Telease_ contentphp': 刷新 父 框架 页 release_content.php 中 的 信息 。 


1.12 付费 信息 管理 模块 设计 


1.12.1 付费 信息 管理 模块 概述 


付费 信息 管理 模块 主要 包括 查看 图 书 列 表 、 付 费 信息 审 核 和 付费 信息 删除 等 3 个 功能 。 付 费 信息 
管理 模块 的 框架 图 如 图 1.69 所 示 。 


付费 信息 管理 


付费 信息 列表 


付费 信息 删除 


图 1.69 付费 信息 管理 模块 的 框架 图 
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1.12.2 ”付费 信息 管理 模块 技术 分 析 


付费 信息 管理 页 面 在 实现 信息 审核 及 删除 的 功能 时 应 用 到 了 UPDATE 更 新 语句 和 DELETE 删 除 语 
句 。 下 面 对 这 两 个 语句 进行 详细 的 讲解 。 

1. UPDATE 语句 

UPDATE 语句 用 来 改变 单行 上 的 一 列 或 多 列 的 值 , 或 者 改变 单个 表 中 选 定 的 一 些 行 上 的 多 个 列 值 。 
UPDATE 语句 的 语法 如 下 : 

UPDATE<table name | view_name> 

SET <column name>=<expression> 


[....,<last column name>=<last expression>] 
[WHERE<search condition>] 


UPDATE 语句 的 参数 说 明 如 表 1.10 所 示 。 


参数 


table name 


View_name 


SET 


column_name 


expression 


表 1.10 UPDATE 语句 的 参数 说 明 
说 了 明 

需要 更 新 的 表 的 名 称 。 如 果 该 表 不 在 当前 服务 器 或 数据 库 中 ， 或 不 为 当前 用 户 所 有 ， 这 个 名 称 
可 用 链接 服务 器 、 数 据 库 和 所 有 者 名 称 来 限定 
要 更 新 的 视图 的 名 称 。 通 过 view_name 来 引用 的 视图 必须 是 可 更 新 的 。 用 UPDATE 语句 进行 的 
修改 ， 至 多 只 能 影响 视图 的 FROM 子 句 所 引用 的 基 表 中 的 一 个 
指定 要 更 新 的 列 或 变量 名 称 的 列表 
含有 要 更 改 数据 的 列 的 名 称 。column name 必须 驻 留 于 UPDATE 子 句 中 所 指定 的 表 或 视图 中 。 
标识 列 不 能 进行 更 新 。 如 果 指 定 了 限定 的 列 名 称 ， 限 定 符 必 须 同 UPDATE 子 句 中 的 表 或 视图 的 
名 称 相 匹配 
变量 、 字 面值 、 表 达 式 或 加 上 括号 返回 单个 值 的 subSELECT 语句 。expression 返回 的 值 将 替换 


column_name 或 @variable 中 的 现 有 值 


WHERE 


指定 条 件 来 限定 所 更 新 的 行 。 根 据 所 使 用 的 WHERE 子 句 的 形式 ， 有 两 种 更 新 形式 : 
搜索 更 新 指定 搜索 条 件 来 限定 要 删除 的 行 
定位 更 新 使 用 CURRENT OF 子 句 指定 游标 。 更 新 操作 发 生 在 游标 的 当前 位 置 


<search_condition> 


为 要 更 新 行 指定 需 满足 的 条 件 。 搜 索 条 件 也 可 以 是 联接 所 基于 的 条 件 。 对 搜索 条 件 中 可 以 包含 
的 谓词 数量 没有 限制 


名 注意 ; 一 定 要 确保 不 要 忽略 WHERE 子 句 ， 除 非 想 要 更 新 表 中 的 所 有 行 。 

下 面 应 用 UPDATE 语句 将 指定 职员 的 工资 进行 调整 ， 例 如 ， 将 “小 璇 ”的 基本 工资 由 2600 元 修 
改 为 3000 元 。 其 SQL 语句 如 下 : 

update tab laborage set jbez=3000 where name=' 小 璇 ' 

2. DELETE 语句 

DELETE 语句 实现 删除 数据 记录 。DELETE 语句 的 语法 如 下 : 


‘Ss1 
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DELETE FROM <table name> 
[WHERE<search condition>] 


DELETE 语句 的 参数 说 明 如 表 1.11 所 示 。 
表 1.11 DELETE 语句 的 参数 说 明 
参数 说 了 明 
可 选 的 关键 字 , 可 用 在 DELETE 关键 字 与 目标 table_name、view_name 或 rowset_function limited 


之 间 
table name 是 要 从 其 中 删除 行 的 表 的 名 称 


FROM 


<search_condition> | 指定 删除 行 的 限定 条 件 。 对 搜索 条 件 中 可 以 包含 的 谓词 数量 没有 限制 


下 面 应 用 DELETE 语句 删除 “职员 姓名 = 小 璇 ”员工 的 基本 信息 。 其 SQL 语句 如 下 : 
DELETE tab_staffer WHERE ygname=' 小 璇 ' 


1.12.3 ”付费 信息 显示 的 实现 过 程 


国 ”付费 信息 显示 页 面 使 用 的 数据 表 : tb_leaguerinfo 

管理 员 在 后 台 功 能 导航 区 的 付费 信息 显示 方式 栏 中 选择 相应 的 信息 类 别 ， 然 后 按 “ 已 付费 ”、“ 未 
付费 ”或 “全 部 ”中 的 任意 一 种 状态 对 付费 信息 进行 管理 。 例 如 ， 在 “信息 类 别 ” 下 拉 列 表 框 中 选择 
“公寓 信息 ”， 在 付费 状态 中 选中 “未 付费 ” 单 选 按钮 ， 单 击 “ 检 索 ” 按 钮 提交 表单 ， 程 序 将 按 指定 
条 件 显 示 出 符合 条 件 的 所 有 信息 ， 运 行 结果 如 图 1.70 所 示 。 


本 更 


四 二 现在 的 省: 九 九 及 供 字 全 息 同 》 后 合营 理 系 二 


时 多 费 信 息 显示 方式 


六 刘 术 人 才 | 
EB 人 所 校 全 
位 于 长 江 落 示 行 责 附 近 ， 新 开业 ， | 
8 位置 侍 ， 装 个 剖 华 , 设 往 弄 全 , 设 
2 EI ee et 于 二。 lostbexsrr。 oorieog | sosotrat 未 机 二 /8 
子 全高 。 折 ， 以 小 枯 可 敢 多 红 提 内 和 尖 ， 价 
加 材 密 信息 显示 方式 :可靠 ， 同 扒 人 床 情 化 莹 
,所 二 音 身 文子 入住 
太守 要 人 要 | | 
Oh i A si A 
。 章 宰 十 厅 ,万 重 起 后 . 有 张 可 
于 二 讽 ， 新 地 消去 本 ,开头 拒 ， 鲍 只 晤 
re 后 末 | ed 和 | 
昌 企业 广告 显 示 方 式 | 
克基 攻关 二 一 -全 硬 位 于 洛阳 | ee 
人 已 扒 大 个 术 推 荐 个 生硬 事 公 击 伍 于 区 了 区 ,基于 全 科 。 | 
Lv 文子 公 他， 现 入 信人 员工 作 和 高 了 四 有 
- 作 委 二 洲 妈 防 入 住 。 有 十 自 里 可 文士 043L-B40Tevey 2007-12-20 2009-01-04 未 付费 证 合 / 卓 陆 


RR 


me 
ER 
Ra 


碍 入 度 供求 信息 同 vee ycll0 co 版 权 所 有 有 


图 1.70 付费 信息 显示 页 面 的 运行 结果 
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本 系统 提供 了 一 组 单 选 按钮 组 成 的 “付费 状态 ”选项 组 ， 付 费 状态 分 为 已 付费 、 未 付费 和 全 部 3 


个 选项 。 选中 “未 付费 ” 单 选 按 钮 ， 则 传递 的 值 为 <0”; 选中 “已 付费 ” 单 选 按钮 ， 则 传递 的 值 为 “1”; 
选中 “全 部 ” 单 选 按 钮 ， 则 传递 的 值 为 “all”。 还 提供 了 一 个 下 拉 列 表 框 ， 供 用 户 选 择 信 息 类 别 。 将 
这 些 单 选 按 钮 与 下 拉 列 表 框 都 在 一 个 表单 中 实现 ， 这 样 ， 当 单 击 “ 检 索 ” 按 钮 提交 表单 后 ， 选 择 的 状 
态 会 通过 表单 进行 传递 。 其 表单 代码 如 下 : 


例 程 15 代码 位 置 光盘 \TM\01\99pursey\admin\left.php 


© <form name="form3" method="post" action="find mianfeiphp" target="mainFrame"> 


<tr> 
<td height="65" align="center"> 
<fieldset style="height:60;width:210"> 
<legend> 友 审核 状态 </legend> 
<input name="state" type="radio" class="inputl" value="1"> 已 审核 
<input name="state" type="radio" class="inputl" value="0" checked> 未 审核 
<input name="state" type="radio" class="inputl" value="all"> 全 部 
</fieldset> 
<td> 
</t> 
<t> 
<td height="34" align="center" background="images/info d.gif> 信 息 类 别 : 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 </option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 </option> 


</select> 
<input type="submit" name="Submit" value=" 检 索 "> 
<td> 
</a> 
</form> 


< 人 代码 贴 二 


素 ， 


@ target: 指定 链接 的 目标 窗口 ，mainFrame 为 内 容 显 示 区 的 框架 名 称 。 另 外 ，target 有 4 个 选项 值 : 

回 _blank: 指定 将 链接 的 目标 文件 加 载 到 未 命名 的 新 浏览 器 窗口 中 。 

回 ”_parent: 指定 将 链接 的 目标 文件 加 载 到 包含 链接 的 父 框架 页 或 窗口 中 ， 如 果 和 包含 链接 的 框 不 是 嵌 套 的 ， 则 链接 
的 目标 文件 加 载 到 整个 浏览 器 窗口 中 。 

_self 指定 将 链接 的 目标 文件 加 载 到 链接 所 在 的 同一 框架 或 窗口 中 。 

回 _top: 指定 将 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 ， 并 由 此 删除 所 有 框架 . 

@ <fieldset><legend>...</legend></fieldset> 标 签 : 在 字符 集 包 含 的 文本 和 其 他 元 素 外 面 绘制 一 个 方 框 。 该 元 素 是 块 元 

必须 成 对 出 现 。 需 要 注意 的 是 ，fieldset 必须 用 在 form 表单 里 ， 一 个 表单 可 以 有 多 个 <fieldset>.…</fieldset>， 每 对 


<fieldset>.…</fieldset> 为 一 组 ， 每 组 的 内 容 描述 使 用 <legend> 设 置 标题 名 称 。 


提交 表单 信息 到 find_mianfei.php 页 ,程序 将 按 管理 员 选 择 的 指定 条 件 显 示 出 符合 条 件 的 所 有 信息 。 


如 果 管理 员 选 中 “全 部 ” 单 选 按钮 ， 那 么 系统 的 代码 如 下 : 


例 程 16 ”代码 位 置 : 光盘 \TMVW1\99purseyvadminfind fufeiphp 


<?php 
include("../conn/conn .php"): // 连 接 数 据 库 文件 
Sstate=$_POST[payfor]: // 获 取 管 理 员 选择 的 付费 状态 
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S$type=$_POST[select]: // 获 取 管理 员 选 择 的 信息 类 别 


/如 果 管 理 员 选 择 的 付费 状态 为 “全 部 ”， 则 查询 管理 员 指 定 信息 类 别 下 的 所 有 付费 信息 ， 并 按 id 降 序 排列 
if(Sstate—"all"){ 
S$sql=mysql_query("select* from tb leaguerinfo where type='$type' order by id"): 
hi 
else{ 
// 如 果 管 理 员 选 择 的 付费 状态 为 “已 付费 ”或 “未 付费 ”状态 ， 则 按 管 理 员 指定 的 条 件 进 行 查询 ， 并 按 id 降 序 排列 
S$sql=mysql query("select* fiom tb leaguerinfo where type='$type' and checkstate=$state order by id"); 


Sinfo=mysql_fetch_array($sql): // 执 行 SQL 语 句 
?> 
// 省 略 供求 信息 标题 的 HTML 代码 部 分 
<2php 
ifSinfo){ // 如 果 检 索 到 了 查询 记录 
dof /应 用 do…while 循 环 语句 输出 付费 信息 
这 Sinfo[checkstate] 一 1){ /如果 付费 状态 的 值 为 1 
$statel=" 已 付费 "; // 则 将 “已 付费 ” 赋 给 变量 $statel 
}else{ // 如 果 付 费 状态 的 值 为 0 
Sstatel=" 未 付费 ": // 则 将 “未 付费 ” 赋 给 变量 $statel 
> 
到 输出 指定 符合 查询 条 件 的 付费 信息 --------------------------------- -- > 


<tr bgcolor="#FFFFFF"> 

<td>&nbsp:<?php echo $infoftitle]:?></td> 

<td width="204">&nbsp;<?php echo $info[content]:?></td> 

<td>&nbsp:<?php echo $info[linkman]:?></td> 

<td>&nbsp:<?php echo Sinfo[tel]:?></td> 

<td>&nbsp;<?php echo S$info[sdate]:?></td> 

<td>&nbsp;<?php echo $info[showday]:?></td> 

<td align="center" class="stylel1"><?php echo $statel:?></td> 

<td align="center" bgcolor="#FFFFFF"> 
<a href="statefu_ok.php?id=<?php echo $info[id]:?>&type=<?php echo $type:;?>&state=<?php echo $state;?>"> 审 核 </a> 
/<a href="fudel_ok.php?id=<?php echo $info[id]:?>&type=<?php echo $type:?>&state=<?php echo $state:?>"> 删 除 </a> 
</td> 


<2php 
}while(Sinfo=mysql_fetch_array($sqD): /do…while 循 环 语句 结束 
// 并 条 件 语句 结束 


// 如 果 未 检索 到 与 管理 员 指 定 条件 相 匹配 的 记录 ， 则 输出 相关 的 提示 信息 
else 
二 { 
<tr align="center" bgcolor="#FFFFFF"> 
<td colspan="8"> 对 不 起 ， 您 检索 的 信息 不 存在 ! </td> 

</t> 
<?php 
4 


?> 
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说明: 上 面 代码 中 省 略 了 分 页 显示 的 代码 的 部 分 ， 限 于 篇 幅 ， 参 见 本 书 附 赠 光 盘 。 
1.12.4 ”付费 信息 审核 的 实现 过 程 


国 ”付费 信息 审核 使 用 的 数据 表 : tb_leaguerinfo 

经 过 审核 的 信息 说 明 该 信息 为 已 付款 信息 。 如 果 企 业 或 个 人 用 户 已 登录 供求 信息 但 未 直接 付费 ， 
想 要 后 期 付款 ， 那 么 管理 员 进 行 审核 时 ， 信 息 不 能 通过 ， 要 求 必须 进行 付款 ， 经 过 审核 的 信息 即 可 在 
前 台 进 行 显示 。“ 审 核 ” 超 链接 的 代码 如 下 : 

例 程 17 ”代码 位 置 : 光盘 \TMW1\99purseyvadminfind fufeiphp 

<a href="statefu_ok.php?id=<?php echo $info[id]:?>&type=<?php echo $type:?>&state=<?php echo $state:?>"> 审 核 </a> 

管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 statefu_ok.php， 用 UPDATE 语句 将 付费 状态 设置 为 1， 说 明 该 信息 已 经 付款 ， 数 据 处 
理 页 的 代码 如 下 : 

例 程 18 代码 位 置 : 光盘 \TMW1\99purseyvadmin'statefu_ok php 


<?php 
include("../conn/conn .php"); // 连 接 数 据 库 文件 
$id=$_GET[id]: // 获 取信 息 id 的 值 
$type=$_GET[type]; 1/ 获取 信息 类 型 
Sstate=$_GET[state]: // 获 取信 息 付费 状态 

@ $sql=mysql query("update tb_leaguerinfo set checkstate=1 where id=$id"); /更 新 对 应 付费 信息 的 状态 为 已 付款 
ifSsqD{ // 如 果 更 新 操作 成 功 

@ echo "<script>alert(' 该 信息 已 经 通过 审核 ! 

"):window.location.href='find fufei.php?type=Stype&state=Sstate':</script>": 1// 弹 出 操作 成 功 提示 信息 
} 
else{ // 如 果 更 新 操作 失败 

echo "<script>alert(' 该 信息 审核 操作 失败 ! ):history.backO:</script>": /弹出 操作 失败 信息 

} 
> 

Ah) 代码 贴 二 


@ update.…set: 用 来 修改 指定 表 中 的 数据 。 UPDATE 语句 的 使 用 方法 参见 1.12.2 节 。 

@ type=$type&state=$state: 将 变量 type 与 state 的 值 重新 传 到 find fufei.php 页 ,目的 是 为 了 使 fnd fnfeiphp 页 中 
的 $type 和 Sstate 变量 重新 获得 信息 类 型 和 付费 状态 值 (管理 员 选 择 的 检索 条 件 ) ， 从 而 返回 到 付费 信息 管理 页 ， 并 更 新 
数据 信息 。 如 果 数据 处 理 页 不 对 这 两 个 变量 进行 传 值 ， 那 么 在 返回 到 付费 信息 管理 页 find fufeiphp 页 时 将 会 因为 检索 
不 到 变量 的 值 而 出 错 。 


1.12.5 ”付费 信息 删除 的 实现 过 程 


国 ”付费 信息 删除 使 用 的 数据 表 : tb_leaguerinfo 
付费 信息 管理 页 中 “删除 ” 超 链接 的 代码 如 下 : 
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例 程 19 代码 位 置 : 光盘 \TM\01\99pursey\admin\find fufeiphp 

<a href="fudel ok.php?id=<?php echo $info[id]:?>&type=<?php echo $type:?>&state=<?php echo $state:?>"> 删 除 </a> 

管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 fndel ok php， 用 DELETE 语句 将 id 指定 的 供求 信息 删除 ， 数 据 处 理 页 的 代码 如 下 : 

例 程 20 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\fudel ok.php 


<?php 

include("../conn/conn.php"); // 连 接 数 据 库 文件 

$id=$_GET[id]; // 获 取信 息 id 的 值 
$type=$_GET[typel: // 获 取信 息 类 型 

$state=$ GET[state]: // 获 取信 息 付费 状态 

$sql=mysql query("delete from tb_leaguerinfo where id=$id"); /删除 对 应 的 供求 信息 

if($sqD){ /如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 


echo "<script>alert(' 该 信息 已 经 删除 ! "):window.location.href~'find_fufei.php?type=Stype&state=Sstate';</script>"; 
} 
else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 该 信息 删除 操作 失败 ! '):history.backQ:</script>"; 
} 


?> 


名 注意 : 在 删除 数据 后 ， 仍 需要 将 变量 Stype 和 S$state 的 值 重新 传递 给 付费 信息 管理 页 find_fufeiphp 
页 ， 目 的 是 返回 到 付费 信息 管理 页 ， 查 看 执行 删除 操作 后 的 状态 。 


1.12.6 单元 测试 


在 开发 完 管理 员 模块 后 ， 对 该 模块 进行 单元 测试 。 当 管理 员 对 付费 供求 信息 进行 审核 后 ， 审 核 操 
作成 功 了 ， 但 却 弹出 如 图 1.71 所 示 的 错误 提示 。 


没有 检索 到 数据 信息 的 错误 提示 


"eR 


not a valid MySQL result resource in 
3 php on line 16 
加 免费 信息 显示 方式 


ing: nysql_fetch_arrayO: is not a valid MySQL result resource in 


nd 
supplied argunent 
六 审核 状态 ee \AppServ\wrw\IN\99Pursey\adnin\find fufei. ?hp on line 37 
已 审核 人 未 审核 个 | 因 您 现在 的 位 置 : 九 九 度 供求 信息 网 》 后 台 和 理 系统 


信息 类 别 : [ 求 阴 信息 ” 菩 ] 弄 于 | 


加 付费 信息 显示 方式 


太 付费 状态 人 
个 已 人 费 人 未 付 费 个 全 部 
| 信息 类 别 为 空 ， 说 明 没有 获取 到 值 


信息 类 别 ; [- 公 寅 信息 。 习 ] 区 大 | 


图 1.71 审核 付费 供求 信息 的 错误 提示 
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在 图 1.71 中 的 错误 提示 中 可 以 看 出 ， 在 付费 信息 管理 页 的 第 16 行 和 第 17 行 出 现 了 问题 。 下 面 看 
一 下 出 现 问题 的 这 两 行 代码 : 


这 Sstate 一 "all){ /如果 管 理 员 选择 的 付费 状态 为 “全 部 ”， 则 执行 下 面 的 SQL 语句 
$sql1=mysql query("select count(*) as total from tb leaguerinfo where type='Stype' order by id"); 
jelse{ /否则 ， 执 行 下 面 的 SQL 语句 


$sql1=mysql query("select count(*) as total from tb leaguerinfo where type='$type' and checkstate=$state order by 
1: 
DY A bse 


$minfo=mysql fetch array($sql]): 


/这 素 半 束 太宗 训 站 训 站 宗 灾 兴 寂 灾 站 灾 汪 站 灾 汪 中 全 ] 了 和 本人 和 2 本 二 员 宙 中 本 机 本 本 二 内 机 中字 机 中字 机 中 字 可 员 字 守 吕 可 人 


Stotal=Sminfo[total]: 


从 代码 中 可 以 看 出 ，SQL 语句 的 书写 并 没有 错误 。 图 1.71 所 示 的 页 面 的 结果 显示 当前 信息 类 别 为 
空 ， 则 说 明 管 理 员 选择 的 信息 类 别 没有 传 过 来 值 ， 由 此 可 以 看 出 ， 这 是 由 于 在 执行 审核 后 页 面 重 新 刷 
新 了 ， 因 此 检索 不 到 管理 员 选 定 的 查询 条 件 值 。 

“审核 ” 超 链接 的 源 代码 如 下 : 


例 程 21 代码 位 置 : 光盘 \TM\01\99pursey\admin\find fufeiphp 
<a href="statefu_ok.php?id=<?php echo $info[id]:?>"> 审 核 </a> 


审核 处 理 页 的 源 代码 如 下 : 

例 程 22 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\statefu_ok.php 

<?php 

$id=$_ GET[id]: /获取 信息 id 的 值 

$sql=mysql query("update tb_leaguerinfo set checkstate=1 where id=$id"): /更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sqD){ // 如 果 更 新 操作 成 功 ， 弹 出 提示 信息 


echo "<script>alert(' 该 信息 已 经 通过 审核 ! "):window.location.href='find_fufei.php;</script>": 
} 


解决 该 问题 的 方法 需要 在 “审核 ” 超 链 接 传 值 时 将 管理 员 选 定 的 “信息 类 型 ”和 “审核 状态 ”的 
变量 值 一 同 传递 到 数据 处 理 页 ， 当 审核 操作 完成 后 ， 再 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重新 
传递 给 付费 信息 管理 页 find_fufei.php 即 可 。 

“审核 ” 超 链接 的 修改 后 的 代码 〈 加 粗 的 代码 部 分 为 修改 的 代码 部 分 ) 如 下 : 

例 程 23 ”代码 位 置 : 光盘 VTMVW1\99purseyvadminfind fufeiphp 


<a href="statefu_ok.php?id=<?php echo $info[id]:?>&type=<?php echo Stype;?>&state=<?php echo Sstate;?>"> 审 核 
</a> 


总 技巧: 在 传递 多 个 变量 时 ， 变 量 之 间 用 “&” 符 号 分 隔 。 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 statefu_ok.php， 在 执行 完 更 新 操作 后 ， 需 要 将 “信息 类 型 ” 和 “和 审核 状态 ”的 变量 值 
重新 传递 到 付费 信息 管理 页 find_fufeiphp， 加 粗 的 代码 部 分 为 修改 的 代码 部 分 。 
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例 程 24 ”代码 位 置 : 光盘 \TMWO1\99purseyvadminstatefu okphp 


<2php 
$id=$_GET[id]: /获取 信息 id 的 值 
Stype-S_GET[type]; // 获 取信 息 类 型 
Sstate=$ GET[state]; // 获 取信 息 付 费 状态 
$sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id"); 。 “”// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sqD{ // 如 果 更 新 操作 成 功 
echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
"):window.location.href='find fufei.php?type=Stype&state=Sstate';</script>"; 1/ 弹出 操作 成 功 提示 信息 


1.13 网 站 发 布 


开发 九 九 度 供求 信息 网 站 的 最 终 目的 是 将 其 发 布 到 Intermet 上 ， 供 用 户 浏览 访问 。 

在 服务 器 上 上 传 网 站 ， 首 先 需 从 服务 商 处 申请 固定 的 了 P 号 ， 然 后 再 注册 一 个 域名 ， 并 将 域名 指定 
到 该 他 地 址 。 在 服务 器 上 安装 PHP 的 开发 环境 ， 最 后 上 载 网 站 。 

下 面 以 “ 九 九 度 供求 信息 网 ”为 例 ， 来 讲解 在 个 人 服务 器 上 上 传 网 站 的 过 程 。 

(1) 右 击 “ 网 上 邻居 ”， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 打 开 “ 网 络 连接 ”对 话 框 。 

(2) 右 击 “ 本 地 连接 ”， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ,打开 “本 地 连接 属性 ”对 话 
框 ， 如 图 1.72 所 示 ， 在 “此 连接 使 用 下 列 选 定 的 组 件 ” 列 表 框 中 选中 “Internet 协议 〈TCP/IP) ” 复 选 
框 ， 然 后 单 击 “ 属 性 ”按钮 ， 按 服务 商 提供 的 指定 亿 设置 服务 器 的 卫 地址 ， 如 图 1.73 所 示 。 


上 本 地 连接 尾 性 XxX| Internet 协议 (TCP/IP) 尾 性 21xl 
常规 “| 身价 验 评 | 高 甸 | 
过 接 时 使 用 
加 Realtek RTLB139 (人 ) PCI Fast Ethernet Adapter 
卫 置 已 


口 
回 轧 mierosoft 网 络 的 文件 和 打印 机 共享 
回信 Internet 协议 CTCF/IP) 


安装 四. Ew __] 属性 


说 明 
| 了 和 i 它 提供 跨越 多 种 互联 网 络 


拒 连接 后 在 通知 区 域 显示 图 标 外 ) 


图 1.72 “本 地 连接 属性 ”对 话 框 1.73 “Internet 协 议 (TCP/IP) 属性 ”对 话 框 
(3) 安装 PHP 的 开发 环境 ， 参 见 本 章 1.4 节 。 
(4) 将 “ 九 九 度 供求 信息 网 ”上 载 到 服务 器 指定 的 路 径 下 ， 如 图 1.74 所 示 。 
(5) 在 正 浏览 器 地 址 栏 中 输入 申请 的 域名 ， 发 布 的 网 站 运行 结果 如 图 1.75 所 示 。 
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恒 九 九 供求 信息 网 


ET TE ET 


Ee BICEPEOPSE .ch 和 园 " 二 记 
miko) ret Ww testo eon eR 


避 回 a 
要 要 发 布 )( 管理 员 改 录 ) (加入 由 其 | 设 为 首页 | 联 朱 我 们 ) 


多 

入 二 ， 柏 且 # 可 外 名 ,可 月 立 度 斩 4 内 人 均 

为 有 相对 稳定 工作 的 女孩 室内 温 权 ， 安 和 ?专人 管理 * 是 奋 过 局 白领 女 性 的 如 心 家 园 
ie 联系 人 : 完 妇 士 。 联系 电话 : 1s5s8Ssal2w 

办 度 导入 在 贱 开 通 所 有 从 


『 公 富 候 息 】 新 开 高 档 注 争 女子 公认 


图 1.75 “ 九 九 度 供求 信息 网 ”发 布 后 的 运行 结果 


1.14 开发 技巧 与 难点 2 


1.14.1 查询 关键 字 描 红 功 能 
在 九 九 度 供求 信息 网 前 台 信 


信息 检索 过 程 中 体现 了 方便 快捷 的 人 性 化 原则 ， 为 了 方便 浏览 者 查阅 信 
息 ， 便 于 查找 与 浏览 者 的 关键 字 相符 合 的 信息 ， 在 搜索 引擎 中 添加 描 红 功能 。 


未 

查询 关键 字 描 红 是 指 将 查询 关键 字 以 特殊 的 颜色 、 字 号 或 字体 进行 标识 ， 这 样 可 以 使 浏览 者 快速 
找到 所 需 的 关键 字 ， 方 便 浏 览 者 从 搜索 结果 中 查找 所 需 内 容 。 查 询 关 键 字 描 红 适用 于 模糊 查询 。 下 面 
介绍 如 何 实现 查询 关键 字 描 红 。 


中 
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本 系统 用 str_ireplaceO 函 数 来 替换 查询 关键 字 ， 当 显示 所 查询 的 相关 信息 时 ， 将 输出 的 关键 字 的 字 
体 替 换 为 红色 。 描 红 功能 的 业务 流程 如 图 1.76 所 示 。 


用 户 


设置 查询 条 件 


将 关键 字 描 红 


图 1.76 查询 关键 字 描 红 

str_ireplace() 函 数 用 于 将 某 个 指定 的 字符 串 替换 为 另 一 个 指定 的 字符 串 ， 不 区 分 大 小 写 。 该 函数 的 
语法 如 下 : 

mixed str_ireplace ( mixed search, mixed replace. mixed subject [, int &count]) 

该 函数 将 所 有 在 参数 subject 中 出 现 的 参数 search 以 参数 replace 取代 ， 参 数 &count 表示 取代 字符 
串 执行 的 次 数 。 

str_ireplace0) 函 数 的 参数 说 明 如 表 1.12 所 示 。 

表 1.12 str_ireplace() 函 数 的 参数 说 明 


必要 参数 ， 指 定 需要 查找 的 字符 串 
Teplace 必要 参数 ， 指 定 蔡 换 的 值 
Subject 必要 参数 ， 指 定 查找 的 范围 

可 选 参数 ， 获 取 执 行 蔡 换 的 数量 


全 注意 : 该 函数 在 执行 替换 的 操作 时 ， 是 不 区 分 大 小 写 的 。 如 果 需 要 对 大 小 写 加 以 区 分 ， 可 以 使 用 
str_replace() 喜 数 。 


本 系统 应 用 str_ireplace0 函 数 替 换 查询 字符 串 为 红色 的 字符 串 ， 关 键 代码 如 下 : 


<2php 
include("conn/conn.php"): // 连 接 数 据 库 文件 
$content=$_POST[content]: /获取 查询 关键 字 


$sql1=mysql _ query("select * from tb leaguerinfo where checkstate=1 and type= Stype' and content like'9%6$content96 or 
title like'%$content%' or linkman like'%$content%%' or tel like'%6Scontent96”): 

Sinfol=mysql fetch_ array($sql1): // 采 用 模糊 信息 资源 查询 

/下 面 应 用 str_ireplace0O 将 指定 的 查询 关键 字 用 红色 文字 蔡 代 ， 并 输出 替换 后 的 字符 串 

echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>".$infol[type]): // 芝 换 信息 类 型 为 红色 字体 
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echo str_ireplace($content."<font color=#FF0000'>".$content."</font>",$infol[title]): /车 换 信息 标题 为 红色 字体 
echo str_ireplace($content."<font color=#FF0000'>".$content."</font>",$infol[edate]): // 芋 换 发 布 时 间 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$infol[content]); ”// 蔡 换 信 息 内 容 为 红色 字体 
echo str_ireplace($content."<font color=#FF0000'>".$content."</font>",$infol[linkman]); // 蔡 换 联 系 人 为 红色 字体 

echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",S$info]l[tel]): // 蔡 换 联系 电话 为 红色 字体 


1.14.2 ”表单 数据 的 提交 方式 


获取 表单 元 素 提交 的 值 是 表单 应 用 中 最 基本 的 操作 方法 。 表 单数 据 的 传送 方法 有 get 方法 和 post 
方法 两 种 , 通过 <form> 的 method 属性 来 指定 。 下 面 来 具体 讲解 一 下 这 两 种 方法 在 实际 工作 中 的 应 用 范 
围 和 使 用 技巧 。 


1. 通过 get 方法 提交 数据 

使 用 get 方 法 时 ， 表 单数 据 被 当 作 url 的 一 部 分 一 起 传 过 去 。 格 式 为 : 

http://url?namel=valuel &name2=value2...... 

tl: 表单 响应 地 址 。 例 如 ，127.0.0.1/index.php。 

回 name: 表单 元 素 的 名 称 。 例如, <input type="text" name="user">, 这 里 name 的 属性 值 就 是 user。 
通过 name 值 可 以 获取 value 的 属性 值 。 

value: 表单 元 素 的 值 。 例 如 ，<input type="text" name="user" value="mr">， 意 思 是 名 字 叫 user 
的 text 表单 元 素 的 值 为 mr。 

[加 说 明 : url 和 表单 元 素 之 间 用 “?” 隔 开 ， 而 多 个 表单 元 素 之 间 用 “&” 隔 开 ， 每 个 表单 元 素 的 格式 
都 是 “name=value”， 固 定 不 变 。 

PHP 使 用 $_GET 预定 义 变量 自动 保存 通过 get 方法 传 过 来 的 值 ， 使 用 格式 为 : 

$_GET[name] 

这 样 ， 就 可 以 直接 使 用 名 字 为 name 的 表单 元 素 的 值 了 。 


总 技巧， 有 的 PHP 版 本 中 直接 写 Sname 就 可 以 调用 表单 元 素 的 值 ， 这 和 php.ini 文件 的 配置 有 关系 。 
定位 到 GLOBAL=ON/OFF 行 ， 如 果 值 为 ON， 就 可 以 直接 写成 Sname， 反 之 则 不 可 以 。 直 
接应 用 表单 名 称 十 分 方便 ， 但 也 存在 一 定 的 安全 隐患 。 推 荐 读者 关闭 GLOBAL 项 。 


下 面 来 看 一 个 使 用 文本 框 传 值 的 程序 ， 程 序 中 包含 一 个 文本 框 元 素 。 在 文本 框 中 输入 信息 ， 当 单 
击 “ 提 交 ” 按 钮 时 ， 文 本 框 内 的 信息 就 会 和 url 一 起 显示 在 地 址 栏 中 ， 代 码 如 下 : 


<form name="login" method="get" action="index.php"> 


</form> 


总 技巧，get 方法 是 使 用 url 来 进行 传 值 的 ，“ 加 入 收藏 天 ”的 功能 就 是 get 方法 的 应 用 之 一 。 想 要 
将 一 个 带 参 数 的 网 址 加 入 到 “收藏 夹 ” 中 ， 只 能 使 用 url， 它 可 以 脱离 表单 的 束缚 。 文 字 、 
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图 片 等 都 可 以 使 用 这 种 方法 来 传 值 。 

2. 通过 post 方法 提交 数据 

get 方法 有 个 最 大 的 缺点 ， 就 是 它 的 信息 是 显示 在 客户 端 浏览 器 上 的 ， 这 使 用 户 的 资料 暴露 无 遗 ， 
而 且 url 本 身受 长 度 限 制 (1024KB), 不 能 传输 较 大 的 数据 。 这 时 可 以 选择 post 方 法 。 使 用 时 , 将 <form> 
表单 中 的 属性 method 设置 成 post 即 可 。post 方法 不 依赖 url， 所 有 提交 的 信息 在 后 台 传 输 ， 不 会 显示 
在 地 址 栏 中 ， 安 全 性 高 ， 而 且 没 有 长 度 限制 。 

使 用 PHP 的 $ POST[name] 变 量 可 以 获取 表单 元 素 的 值 ， 格 式 和 $_GET[name] 类 似 : 

$_POST[name] 

例如 ， 使 用 的 post 方法 返回 文本 框 信息 ， 代 码 如 下 : 


<form name="login" method="post" action="index.php"> 


</form> 


1.15 MySQL 数据库 技术 专题 


phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 ， 它 是 基于 PHP 语言 编写 的 。 
该 工具 是 B/S 结构 、 基 于 Web 跨 平 台 的 管理 程序 ， 并 且 支 持 简体 中 文 ， 可 以 在 官方 网 站 免费 下 载 。 安 
装 后 在 浏览 器 地 址 栏 输入 http:/127.0.0.1/phpMyAdmin/ 即 可 进入 MySQL 的 管理 界面 。 

phpMyAdmin 为 Web 开发 人 员 提 供 了 类 似 于 Access、SQL Server 的 图 形 化 数据 库 操作 界面 ， 通 过 
该 管理 工具 可 以 进行 绝 大 部 分 的 MySQL 操作 ， 包 括 对 数据 库 及 数据 表 的 建立 和 维护 。 


1.15.1 创建 和 删除 数据 库 


1. 创建 数据 库 

在 phpMyAdmin 的 主 界面 中 有 两 个 文本 框 和 “创建 ”按钮 ， 首 先 在 文本 框 中 输入 数据 库 的 名 称 ， 
然后 选择 编码 , 最 后 单 击 “ 创建 ”按钮 , 这 样 新 的 数据 库 就 可 以 被 创建 成 。 例 如 创建 一 个 名 称 为 db_pursey 
的 数据 库 ， 首 先 在 文本 框 中 输入 db_pursey, 之 后 在 下 拉 列 表 框 中 找到 要 使 用 的 编码 , 在 Windows 下 一 
般 选择 gb2312_chinese_ci， 如 图 1.77 所 示 。 

最 后 单 击 “ 创 建 ” 按 扭 ， 这 样 名 为 db_pursey 的 数据 库 就 被 创建 成 功 ， 执 行 结 果 如 图 1.78 所 示 。 

该 数据 库 名 称 出 现在 左 侧 导 航 栏 的 数据 库 下 拉 菜单 中 ， 选 择 这 个 数据 库 ， 在 右 侧 界 面 中 可 以 对 该 
数据 库 进 行 操作 ， 如 结构 、SQL、 导 出 、 搜 索 、 查 询 、 删 除 等 ， 单 击 相应 的 按钮 即 可 进入 相应 的 操作 
界面 。 但 是 在 创建 数据 库 还 没有 创建 数据 表 的 情况 下 ， 只 能 够 执行 结构 、SQL 和 删除 3 项 操作 ， 其 他 
操作 不 可 以 执行 。 

2. 删除 数据 库 

要 删除 某 个 数据 库 ， 首 先 在 左 侧 的 下 拉 菜单 中 选择 该 数据 库 ， 然 后 单 击 右 侧 界面 中 的 “删除 ” 按 
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钮 即 可 。 
本 localhost 
四 服务 器 版 本 : 50.24a-community-ntlog 
Be » Protocol version: 10 
司 加 回回 区 中 服务 器 : localhost via TCPAP 


， 用户: root@localhost 

国 MySQL 字符 集 UTF-8 Unicode (utfe) 

辆 MysaL 连接 校对 : [962312_chinese_ci 司 @ 
若 创 建 一 个 新 的 数据 库 回 


i re 习 和 时 
A Oe 
输入 数据 库 名 国 显 示 MySQl 的 系统 交 量 @ 选择 编码 格式 


图 1.77 phpMyAdmin 管理 界面 


可 加 服务 器 :1ocalhost ， 忆 数据 库 : db_pursey 


与 当前 数据 库 和 表 相 关 的 操作 


数据 库 db_pursey 已 经 建立 。 


三 SQL 查询 : 
CREATE DATABASE "db_pursey 


数据 库 中 没有 表 。 输入 新 建 表 名 
广角 在 数据 库 db_p 直 sey 中 创建 一 个 新 家 一 一 一 一 一 一 一 一 | 
F(aomm  ) Number offies (BE —) 

新 表 字段 仆 数 | 


图 1.78 数据 库 的 建立 


1.15.2 ”创建 和 删除 数据 表 


针对 表 级 操作 是 在 选 定 了 数据 库 的 情况 下 进行 的 , 即 表 级 操作 的 前 题 是 用 户 必须 选择 一 个 数据 库 ， 


在 该 数据 库 中 进行 表 的 建立 和 维护 。 
1. 创建 数据 表 


创建 数据 库 db_pursey 后 ， 在 右 侧 页 面 中 会 出 现 如 图 1.78 所 示 的 数据 表 创 建 提示 页 面 ， 完 成 数据 


表 的 创建 操作 。 


首先 在 表单 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “ 执 行 ” 按 钮 ， 进 入 各 个 字段 的 详细 信息 录 
入 表单 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 和 索引 的 类 型 等 。 在 这 里 就 完成 了 对 
表 结构 的 详细 设置 ， 如 图 1.79 所 示 。 


当 所 有 的 信息 都 输入 以 后 ， 就 可 以 单 击 “ 保 存 ” 按 钮 ， 成 功 创建 数据 表 tb_admin。 


一 个 新 的 数据 


表 被 创建 后 ， 进 入 到 数据 表 页 面 中 ， 在 这 里 可 以 通过 改变 表 的 结构 来 修改 表 ， 可 以 执行 添加 新 的 列 、 
删除 列 、 索 引 列 、 修 改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 操作 ， 如 图 1.80 所 示 。 
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图 服务 器 :localhost 》 部 数据库: db_pursey ， 国 表 :tb_admin 
有 & 类 型 加 长 度 荐 1 可 于 ull 要 外 


图 图 
加 Im 司 『 I 司 [ranw 司 [auo_ meemen 司 ec C 
Fame [vacma 3 [0 [ov2312 chinese ct 本 | [notnun BJT Fs|s 
rm | Cr | ST | ET | Cr | | | 


Er 


图 1.79 创建 表 信 息 


胃 服务 器 : localhost 》 曲 数据 库 : db_pursey ， 国 表 :tb_admin 
_ 国 浏 览 ” 因 到 枸 .及 sal 万 搜索 和 白 入 ”并 导出 并 Import 多 操作 


—aoto-inereme 图 
厂 bookname varchar(50) gb: gsecrr 访 [esa Ee 2 
厂 price varchar(50) gb2312_chinese_ci Lr | 
人 全 选 /全 部 不 选 大 思 哆 围 儿 X 圈 园 图 贺 
候 打 印 预览 别 关 系 查看 局 规划 表 结构 @ 


图 1.80 ”操作 列表 


2. 删除 数据 表 


执行 删除 表 的 操作 很 简单 ， 只 要 单 击 图 1.80 所 示 页 面 上 方 的 “删除 ”按钮 ， 就 可 以 轻松 地 删除 当 
前 数据 表 。 


本 章 依据 软件 开发 流程 介绍 了 九 九 度 供求 信息 网 的 开发 过 程 。 在 开发 任何 一 个 项 目前 ， 首 先 要 充 
分 做 好 前 期 准备 ， 如 完善 的 需求 分 析 、 清 晰 的 业务 流程 、 合 理 的 程序 结构 等 ， 这 样 ， 在 后 期 的 程序 开 
发 中 才 会 得 心 应 手 , 有 备 而 无 患 。 通 过 本 章 的 学 习 , 读者 可 以 了 解数 据 库 建 模 的 概念 , 掌握 PowerDesigner 
数据 库 建 模 的 方法 ， 熟 悉 框架 技术 在 Web 应 用 程序 中 的 应 用 。 


第 章 


BCTY365 网 上 社区 


(Apachet PHP+phpMyAdmin+MySQL 5.0 实现 ) 
(器 " 视频 讲解 : 2 小 时 24 分钟 ) 


随 着 人 类 文明 的 不 新 进步 ,网络 这 个 虚拟 世界 也 在 发 生 着 变化 ， 现 实 世界 中 的 
所 有 内 容 在 网 络 上 都 有 反映 。 现 实 世 界 中 物 以 类 聚 ， 人 以 群 分 网 络 世界 中 也 形成 
了 各 式 各 样 的 社区 。 这 些 社 区 建立 在 将 殊 的 兴趣 点 (如 体育 、 新 闻 、 宠 物 、 电 影 和 
游戏 等 ) 、 将 殊 的 人 群 、 将 殊 的 爱好 或 者 是 将 殊 的 服务 上 等 。 社 区 也 是 一 种 将 定 的 
商业 模式 ， 它 本 身 可 以 进行 电子 商务 ， 也 可 以 完全 与 商务 无 关 。 

所 谓 网 上 社区 是 指 包 括 BBS 论坛 、 聊 天 室 、 博 客 等 形式 在 内 的 网 上 交流 空间 ， 
同一 主题 的 网 上 社区 集中 了 有 具有 共同 兴趣 的 访问 者 ， 由 于 有 众多 用 户 的 参与 ,因此 
具备 了 交流 的 功能 ， 成 为 一 个 营销 场所 。 

网 上 社区 具有 各 种 不 同 的 表现 形式 和 规模 ， 有 个 人 创办 的 社区 ， 功 能 和 界面 追 
求 时 尚 、 个 性 突出 ;) 有 大 型 的 商业 性 质 社 区 ， 以 盈利 为 目的 ， 分 类 多 元 化 ， 适 合 不 
同类 型 的 网 民 。 

本 章 开发 的 BCTY365 网 上 社区 主要 面向 程序 开发 人 员 ， 集 论坛 、 留 言 板 、 软 
件 下 载 、 升 级 下 载 、 技 术 支 持 和 在 线 购物 等 功能 于 一 身 ， 既 是 一 个 程序 开发 者 交流 
的 平台 ， 更 是 一 个 网 络 营销 的 场所 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


WY 网 上 社区 开发 的 基本 过 程 MY 在线 论坛 功能 的 实现 方法 

HI” 如何 做 需求 分 析 和 系统 设计 WI 在 Linux 操作 系统 下 搭建 PHP 开发 环境 
”如 何 设 计 和 创建 数据 库 、 数 据 表 训 ”在 Linux 操作 系统 下 发 布 网 站 

WI 如何 设计 公共 类 WY 在 线 支 付 技 术 

MW 软件 上 和 传 和 下 载 功 能 的 实现 方法 
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2.1 开发 背景 


随 着 市 场 竞争 的 日 益 激 烈 ， 企 业 的 生存 和 发 展 之 路 更 加 艰难 ， 要 想 使 企业 保持 旺盛 的 生命 力 ， 企 
业 必 须要 跟 上 时 代 发 展 的 脚步 ， 不 断 注入 新 的 活力 。 某 科技 公司 为 适应 市 场 的 需求 ， 增 加 公司 在 互联 
网 上 的 影响 力 ， 将 开发 一 个 网 上 社区 系统 ， 为 广大 的 编程 爱好 者 提供 一 个 交流 的 平台 ， 并 且 以 此 来 推 
广 该 公司 的 软件 产品 。 


2.2 系统 分 析 


当 一 个 开发 项 目 被 确立 时 ， 首 先 要 做 的 就 是 需求 分 析 、 可 行 性 分 析 ， 然 后 编写 项 目 计划 书 ， 以 使 
项 目 开发 人 员 了 解 和 掌握 网 站 的 前 期 策划 和 网 站 开发 流程 。 


2.2.1 需求 分 析 


在 开发 网 上 社区 之 前 ， 首 先 要 明确 所 要 开发 的 社区 属于 什么 类 型 ， 是 个 人 的 社区 系统 ， 还 是 商业 
化 的 社区 系统 ， 并 且 要 知道 开发 的 社区 是 面向 什么 样 的 人 群 ， 是 普通 网 民 ， 还 是 专业 的 技术 人 员 ， 或 
者 是 其 他 的 特殊 群体 。 针 对 不 同 的 人 群 ， 社 区 应 该 具有 不 同 的 特点 。 当 明确 了 这 些 ， 项 目 开发 的 思路 
就 清晰 了 ， 然 后 再 对 网 站 上 一 些 相关 的 社区 进行 考察 、 分 析 ， 从 中 吸取 经 验 ， 并 结合 企业 的 要 求 以 及 
实际 的 市 场 调查 结果 ， 提 出 一 个 合理 的 网 上 社区 网 站 功能 架构 。 本 网 站 需求 如 下 : 
网 站 设计 页 面 要 求 整洁 、 美 观 大 方 ， 能 够 展示 企业 形象 。 
网 站 页 面具 有 banner 广告 ， 树 立 企业 良好 的 口碑 宣传 。 
设计 主要 从 编程 者 的 角度 考虑 ， 为 编程 者 解决 在 开发 中 出 现 的 问题 。 
展示 出 企业 全 力 推出 的 软件 产品 和 提供 的 免费 软件 ， 以 此 吸引 浏览 者 。 
提供 一 个 良好 的 网 上 购物 的 操作 平台 。 
提供 技术 支持 ， 解 决 编程 过 程 中 常见 的 问题 。 
提供 一 个 讨论 和 研究 问题 的 平台 。 
做 到 让 广大 浏览 者 关注 企业 的 动态 。 
为 客户 提供 反馈 信息 的 平台 ， 能 够 作 到 及 时 与 客户 进行 沟通 。 
完善 的 后 台 管理 系统 。 


加 


办 办 办 办 办 罗 办 多 罗 


2.2.2 ”可行 性 分 析 


可 行 性 分 析 是 世界 上 普遍 采用 的 一 种 研究 工程 项 目 是 否 可 行 的 科学 。 其 通过 各 种 有 效 的 方法 ， 对 
工程 项 目 进行 分 析 ， 从 技术 、 经 济 、 市 场 等 方面 加 以 评价 ， 最 终 给 投资 决策 者 提供 是 否 选 择 该 项 目 进 
行 开发 的 依据 。 
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BCTY365 网 上 社区 项 目 开发 的 可 行 性 分 析 主 要 从 以 下 两 个 方面 考虑 。 

1. 经 济 可 行 性 分 析 

企业 为 扩大 公司 的 影响 力 ， 特 推出 软件 产品 ， 并 采用 网 上 社区 的 形式 在 网 络 上 进行 推广 ， 这 样 不 
但 可 以 汇聚 更 多 的 人 气 ， 而 且 可 以 让 更 多 的 人 了 解 该 企业 ， 从 而 达到 推广 企业 软件 产品 的 目的 ， 最 终 
为 企业 带 来 更 大 的 收益 。 更 重要 的 一 点 是 采取 该 方法 的 成 本 相对 其 他 的 电视 广告 或 者 人 力 宣传 的 成 本 
要 低 得 多 ， 虽然 周 期 很 长 ， 但 是 却 能 够 取得 长 期 的 收益 。 

2. 技术 可 行 性 分 析 

网 上 社区 系统 的 开发 采用 Apache+PHP+phpMyadmin+MySQL 5.0 来 实现 ， 开 发 软件 都 是 免费 的 ， 
可 以 直接 从 网 上 下 载 , 无 须 支付 任何 费用 。 要 完成 BCTY365 网 上 社区 系统 的 开发 ， 必 须 能 够 配置 PHP 
程序 开发 的 环境 ， 掌 握 在 线 支付 、 购 物 车 和 在 线 论坛 技术 。 


2.2.3 ”编写 项 目 计划 书 


根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 的 项 目 开发 计划 要 求 ， 结 合 单位 实际 情 
况 ， 设 计 项 目 计 划 书 如 下 : 

1. 引言 

(1) 编写 目的 

为 了 保证 项 目 开 发 人 员 按 时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合理 的 顺序 开 
展 工作 ， 现 以 书面 的 形式 将 项 目 开 发 生命 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工 
作 责 任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描述 出 来 ， 作 为 项 目 相 关 人 员 之 间 
的 共识 和 约定 ， 项 目 生命 周期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 

BCTY365 网 上 社区 系统 是 本 公司 与 X X X 科 技 有 限 公司 签 订 的 待 开发 项 目 ， 网 站 性 质 为 信息 服务 
类 型 ， 为 企业 与 客户 、 浏 览 者 和 会 员 之 间 提 供 一 个 技术 交流 平台 ， 并 且 全 力 推 出 企业 的 软件 产品 。 项 
目 周期 为 两 个 月 ， 项 目 背 景 规划 如 表 2.1 所 示 。 


表 2.1 项 目 背景 规划 


项 目 名 称 签订 项 目 单位 项 目 负责 人 项 目 承担 部 门 
甲 方 : XXX 信息 科技 有 限 公司 甲 方 : 赵 经 理 设计 部 门 
BCTY365 网 上 社区 系统 开发 部 门 
乙方 : XXX 网 络 科技 有 限 公司 乙方 : 张 经 理 测试 部 站 
2. 概述 
(1) 项 目 目标 


项 目 目 标 应 当 符 合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。BCTY365 网 上 社 
区 系统 的 项 目 目 标 如 下 : 
BCTY365 网 上 社区 系统 主要 用 于 在 网 络 中 树立 企业 的 形象 ， 为 程序 开发 者 提供 一 个 交流 的 平台 ， 
拉 近 企业 与 客户 、 会 员 和 浏览 者 之 间 的 距离 ， 从 而 达到 推广 企业 软件 产品 的 目的 。 整 个 项 目 需 要 两 个 
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月 结束 ， 然 后 交 给 客户 进行 验收 。 

(2) 产品 目标 与 范围 

一 方面 BCTY365 网 上 社区 系统 能 够 为 企业 节省 大 量 人 力 资源 , 企业 不 再 需要 大 量 的 业务 人 员 去 跑 
市 场 ， 闻 接 为 企业 节约 了 成 本 。 另 一 方面 ，BCTY365 网 上 社区 系统 能 够 收集 海量 编程 问题 的 解决 方案 
和 好 的 建议 ， 将 会 有 大 量 用 户 访问 网 站 ， 有 助 于 提高 企业 知名 度 。 

(3) 应 交付 成 果 

项 目 开发 完成 后 ， 交 付 的 内 容 如 下 : 

回 “以 光盘 的 形式 交付 BCTY365 网 上 社区 系统 的 源 程序 、 网 站 数据 库 文 件 、 系 统 使 用 说 明 书 。 

回 “客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 ， 安 装 PHP 开发 环境 ， 协 助 四 

方 购买 域名 ， 将 开发 的 BCTY365 网 上 社区 系统 发 布 到 互联 网 上 运行 。 
回 “网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 
维护 与 服务 。 

(4) 项 目 验收 方式 与 依据 

项 目 验收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开 发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
试 员 根据 用 户 需求 和 项 目 目标 进行 验收 。 项 目 在 通过 内 部 验收 后 ， 然 后 交 给 用 户 进行 验收 ， 验 收 的 主 
要 依据 为 需求 规格 说 明 书 。 

3. 项 目 团队 组 织 

(1) 组 织 结构 

为 了 完成 BCTY365 网 上 社区 系统 的 项 目 开发 ， 公 司 组 建 一 个 临时 的 项 目 团队 ， 由 项 目 经 理 、 系 统 
分 析 师 、PHP 开发 工程 师 、 网 页 设计 师 和 系统 测试 员 构成 ， 如 图 2.1 所 示 。 


项 目 经 理 


系统 分 析 师 PHP 开发 工程 师 网 页 设计 师 ”系统 测试 员 
图 2.1 项 目 团队 组 织 结构 图 


(2) 人 员 分 工 
为 了 明确 项 目 团队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分 工 表 ， 如 表 2.2 所 示 。 
表 2.2 人 员 分 工 表 
姓 名 所 属 部 门 角 色 工作 描述 
负责 项 目的 审批 、 决 策 的 实施 、 项 目的 前 期 
张 达 明 项 目 开 发 部 | 项 目 经 理 分 析 、 和 策划、 项目 开发 进度 的 跟踪 、 项 目 质 


量 的 检查 
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续 表 
姓 名 技术 水 平 所 属 部 门 角 色 工作 描述 
王 言 辉 “| 高 级 系统 分 析 师 项 目 开发 部 | 系统 分 析 师 负责 系统 功能 分 析 、 系 统 框 架设 计 
潘 攀 高 级 PHP 工程 师 项 目 开 发 部 | PHP 开发 工程 师 | 负责 软件 前 后 台 设计 与 编码 
刘 悦 高 级 美工 设计 师 设计 部 网 页 设计 师 负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 
齐 欣 高 级 系统 测试 工程 师 | 项 目 开发 部 _| 系统 测试 员 对 软件 进行 测试 、 编 写 软件 测试 文档 
2.3 系统 设计 


2.3.1 系统 目标 


根据 对 目前 网 络 上 各 种 社区 的 分 析 和 研究 ， 结 合 本 项 目的 实际 需求 ， 且 在 设计 时 应 该 满足 以 下 
目标 : 


界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 
功能 完善 、 结 构 清 晰 。 

重点 突出 企业 的 软件 产品 。 

及 时 更 新 网 站 公告 。 

及 时 查阅 和 回复 客户 反馈 信息 。 

为 用 户 提供 沟通 和 交流 的 平台 。 

购物 车 模块 的 设计 结构 合理 、 流 程 清晰 。 

购物 结算 功能 设计 符合 逻辑 ， 计 算 准 确 。 

订单 处 理 功能 的 设计 及 时 、 准 确 、 安 全 。 

网 上 支付 功能 的 设计 处 理 好 与 网 上 银行 之 间 数 据 的 传递 ， 确 保安 全 、 可 靠 。 
具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 


2.3.2 系统 功能 结构 


办 办 办 办 办 办 多多 多 办 多 


结合 需求 分 析 和 系统 目标 中 的 内 容 ，BCTY365 网 上 社区 系统 的 功能 结构 已 经 设计 完成 。 为 了 使 读 
者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 BCTY365 网 上 社区 前 台 和 后 台 的 功能 模块 结构 图 。 

BCTY365 网 上 社区 前 台 管理 系统 的 功能 设计 如 图 2.2 所 示 。 

BCTY365 网 上 社区 后 台 管理 系统 的 功能 设计 如 图 2.3 所 示 。 


2.3.3 ”系统 预览 


BCTY365 网 上 社区 系统 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 可 参见 光盘 中 的 
源 程序 。 
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第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


前 台 首 页 如 图 2.4 所 示 , 该 页 面 用 于 展示 本 系统 的 功能 模块 , 突出 企业 的 形象 , 推广 企业 的 软件 产 
品 。 后 台 首 页 如 图 2.5 所 示 ， 该 页 面 用 于 实现 对 编程 词典 、 技 术 支持 、 软 件 升级 、 软 件 试用 等 内 容 的 管理 。 


图 2.4 前 台 首页 (光盘 \TM\02\bcty365\index.php) ”图 2.5 后 台 首页 (光盘 \TM\02\bcty365\ admin\default.php) 
在 线 订 购 模块 的 页 面 效果 如 图 2.6 所 示 , 该 页 面 主要 用 于 展示 本 企业 在 线 推出 的 软件 产品 , 实现 对 


产品 的 在 线 购买 功能 。 软件 下 载 模块 的 页 面 效 果 如 图 2.7 所 示 , 该 页 面 主要 用 于 展示 本 企业 提供 的 免费 
软件 ， 并 且 提 供 下 载 链接 。 


图 2.6 在 线 订购 (光盘 \TM\02\bcty365\ morebccd.php) 图 2.7 软件 下 载 (光盘 \TM\02\bcty365\ rjxz.php) 


社区 论坛 模块 的 页 面 效果 如 图 2.8 所 示 , 该 页 面 主要 用 于 展示 论坛 中 的 各 大 版 块 ,并 且 提供 超 链接 
跳 转 到 对 应 的 版 块 。 后 台 的 登录 页 面 效果 如 图 2.9 所 示 ， 该 页 面 主要 实现 后 台 管 理 员 登录 。 


图 2.8 社区 论坛 (光盘 \TM\02\bcty365\ bbs_index.php) 2.9 ”后台 登录 (光盘 \TMW2\bcty365\vadmin\indexphp) 
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2.3.4 ”开发 环境 


在 开发 BCTY365 网 上 社区 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
。 服务器 端 
操作 系统 : Windows XP Server/Linux (推荐 ) 。 
服务 器 : Apache 2.0。 
PHP 软件 : PHP 5.0。 
数据 库 : MySQL 5.0。 
MySQL 图 形 化 管理 软件 : PhpMyAdmin-2.9.0。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 ， 最 佳 效果 1024X768 像素 。 
客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 


一 


因 办 办 办 多 多 多 


区 国信 


2.3.5 ”文件 夹 组 织 结构 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 
的 分 类 ， 分 别 放置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 可 以 确保 网 站 文件 目录 明确 、 
条 理 清晰 ， 同 样 也 便于 网 站 后 期 的 更 新 和 维护 。 本 案例 的 文件 夹 组 织 架构 规划 如 图 2.10 所 示 。 


用 于 存储 网 站 后 台 管 理 文件 

用 于 存储 编程 词典 的 界面 

用 于 存储 网 站 后 台 使 用 的 Css 文 件 

用 于 存储 网 站 后 台 的 图 片 文件 

用 于 存储 升级 下 载 的 文件 

用 于 存储 可 下 载 的 软件 

用 于 存储 连接 数据 库 的 文件 

用 于 存储 网 站 前 台 使 用 的 Css 文 件 

用 于 存储 网 站 前 台 使 用 的 图 片 文件 
用 于 存储 网 站 论坛 中 上 传 的 图 片 文件 


图 2.10 文件 夹 组 织 结构 
2.4 在 Linux 操作 系统 下 搭建 PHP 开发 环境 


Red Hat Linux 9 是 Linux 众多 版 本 中 比较 大 众 化 的 一 版 。 在 安装 系统 时 ， 如 果 选 择 完全 安装 或 者 
选择 Apache、MySQL、PHP 的 安装 包 ， 则 三 者 将 被 安装 到 系统 中 ， 用 户 只 需 将 Apache 和 MySQL 服 


- 国 
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务 启动 就 可 以 使 用 三 者 ， 非 常 方便 ， 但 是 Apache 和 MySQL 的 版 本 可 能 不 是 很 理想 。 为 了 能 够 创建 一 
个 良好 的 PHP 开发 环境 ,这 里 将 详细 介绍 自行 在 Linux 下 安装 和 配置 Apache2+MySQL5.0+PHP5 的 方法 。 
首先 应 该 到 相关 官方 网 站 下 载 三 者 的 安装 包 : 
httpd-2.0.58.tar.gz 或 更 高 版 本 (http://httpd.apache.org/) 。 
mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 或 更 高 版 本 《http://www.mysql.com/) 。 
php-5.0.0.tar.gz 或 更 高 版 本 (http://www.php.net/》。 
libxml2-2.5.10.tar.gz 或 更 高 版 本 (如 果 读 者 系统 中 的 libxml2 的 版 本 已 经 等 于 或 高 于 该 版 本 可 以 不 
必 下 载 该 安装 包 ) 。 


2.4.1 Linux 下 Apache 的 安装 配置 


首先 将 下 载 的 httpd 安装 包 复制 到 适当 的 位 置 ， 例 如 /usr/local/work 下 〈 如 果 目 录 不 存在 ， 可 以 建 
立 该 目录 ) 。 打 开 Red Hat Linux 9 的 主 菜单 ， 选 择 “ 系 统 工具 ”选项 ， 在 弹出 的 菜单 中 选择 “终端 ” 
命令 ， 将 打开 如 图 2.11 所 示 的 终端 对 话 框 。 

[说明 Linux 下 Apache、MySQL 及 PHP 的 安装 都 是 在 如 图 2.11 所 示 的 终端 命令 窗口 通过 命令 方 
式 实现 的 。 


TO 
TITTIETTRTTIETTECTT 
Treoelecs Ihos! root 


图 2.11 Red HatLinux 9 的 终端 命令 对 话 框 


在 该 对 话 框 中 输入 如 下 命令 进入 work 目录 : 

cd /usr/local/work 

在 work 目录 中 输入 如 下 命令 解压 httpd-2.0.58.tar.gz: 
tar xfz httpd-2.0.58.tar.gz 

进入 解压 后 的 目录 httpd-2.0.58 如 下 : 

cd httpd-2.0.58 

建立 makefile， 并 将 Apache2 安装 到 /usr/local/apache2 目录 下 : 
.configure —prefix=/usr/local/apache -enable-module=so 

开始 编译 : 

make 

开始 安装 到 设置 的 目录 去 : 

make install 


到 此 Apache2 的 安装 工作 完成 ， 可 以 在 每 次 启动 系统 时 通过 如 下 命令 启动 或 重新 启动 Apache2 
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服务 : 


/usr/local/apache2/bin/apachect] start 
/usr/local/apache2/bin/apachect] restart 


打开 浏览 器 ， 在 地 址 栏 中 输入 http://127.0.0.1 或 者 http://localhost， 按 Enter 键 ， 如 果 出 现 如 图 2.12 
所 示 的 页 面 ， 则 说 明 Apache2 安装 成 功 。 


文体 E) 编辑 EE) 查看 WW 转 夭 G) 书签 B) 工具 了 窗 DOW 帮助 H) 


把 ” 多 = 区 -是 恒 乏 hup//iocalhbosViext php 加 | 站 


甬 主 页 | | 对 书签 大 Red Hat Newetk 济 Suppon 阔 Shop 项 pdaucs 的 Tainng 


Fe 说 明 ver 已经 安装 成 功 。 您 可 以 在 这 个 目录 中 增加 内 
， 或 者 把 这 个 页 面 痊 : 


这 不 是 你 想 看 见 的 页 面 吧 ? 
Pe ed 是 因为 网 站 管理 员 改 变 了 这 个 站 点 的 设置 。 如 果 有 疑问 ， 请 咨询 维 
护 此 站 点 的 人 员 。 Apache 软 件 基金 会 , 


即 此 站 点 所 使 用 的 网 站 服务 器 软件 的 开发 者 ， 不 负 
责 此 站 点 的 维护 工作 ， 也 无 法 为 您 解决 设置 上 的 问题 。 


Apache 文档 已 经 包含 在 此 发 行 版 中 。 
您 可 以 在 使 用 Apache 的 网 站 服务 器 上 ， 自 由 地 使 用 下 面 的 图 片 。 感 谢 使 用 Apachel 


> 一 一 一 


11 朋 28 二 
O08:51 


图 2.12 测试 Apache 服务 器 
2.4.2 Linux 下 MySQL 的 安装 配置 


将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 复制 到 /usr/local/work 目录 下 ， 建 立 MySQL 账号 : 
groupadd mysql 

在 组 群 中 加 入 MySQL: 

useradd -g mysql mysql 

进入 local 目录 : 

cd /usr/local 

将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 解压 到 该 目录 

tar xfz /usr/local/work/mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 

考虑 到 MySQL 数据库 升 级 的 需要 ， 所 以 通常 以 链接 的 方式 建立 /usrlocalmysql 目录 : 
ln -s mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz mysql 

进入 MySQL 目录 : 

cd mysql 
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在 /usr/local/mysql/data 中 建立 MySQL 的 数据 库 : 
Scripts/mysql install db —user=mysql 


修改 文件 权限 : 


chown -R root . 

chown -R mysql data 

chgrp -R mysql . 

到 此 MySQL 安装 成 功 。 可 以 通过 在 终端 中 输入 如 下 命令 启动 MySQL 服务 : 
/usr/local/mysql/bin/mysqld_safe -user=mysql & 

启动 MySQL 后 输入 如 下 命令 查看 安装 结果 : 

/usr/local/mysql/bin/mysql —uroot 


如 果 终 端 对 话 框 出 现 如 图 2.13 所 示 的 提示 ， 则 说 明 MySQL 安装 成 功 。 


ET 

[root@localhost root]# /usr/local/nysql/bin/nysql -uroot -pl23 
Wlcone to the MSGL nonitor. Commnds end with ; or \g 

your MSQL connection id is 1 to server version: 5.0.0-alpha-standard 


[ype ‘help:’ or “\h’ for help. Type '\e’ to clear the buffer. 


sop 


图 2.13 测试 MySQL 
2.4.3 Linux 下 PHP 的 安装 配置 


首先 查看 系统 中 libxml2 的 版 本 号 ， 如 果 libxml2 的 版 本 号 小 于 2.5.10， 则 需要 安装 
libxml2-2.5.10.tar.gz 或 更 高 版 本 ， 因 为 PHP5 必须 在 libxml2 的 版 本 大 于 2.5.10 的 前 提 下 才能 够 安装 。 
将 libxml2-2.5.10.tar.gz 复制 到 /usr/local/work 目录 下 ， 并 进入 该 目录 : 


cd /usr/local/work 
解压 libxml2-2.5.10.tar.gz: 
tarxfzlibxml2-2.5.10targz 
进入 该 目录 
cd libxml2-2.10 
建立 makefie 并 将 libxml2 安装 到 /usrlocallibxml2 下 : 
_/configure -prefix=/usr/locallibxml2 
始 编译 : 
make 
开始 安装 到 设置 的 目录 去 : 
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make install 

到 此 libxml2 安装 成 功 。 

将 php-5.0.0.tar.gz 复制 到 /usrlocal/work 目录 下 ， 并 进入 该 目录 ; 
cd /usr/local/work 

解压 php-5.0.0.tar.gz: 

tar xfz php-5.0.0.tar.gz 

进入 php-5.0.0 目录 : 

cd php-5.0.0 

建立 makefile: 


.configure —with-apxs2=/usr/local/apache2/bin/apxs \ 
--with-mysql=/usr/local/mysql \ 
--with-libxml-dir=/usr/local/libxml2 


开始 编译 : 
make 
开始 安装 : 
make install 
复制 php.ini-dist 或 php.ini-recommended 到 /usr/local/lib 目录 ， 并 命名 为 php.ini: 
cp php.ini-dist /usr/local/lib/php.ini 
更 改 httpd.conf 文件 相关 设置 , 该 文件 位 于 /usr/local/apache2/conf 中 。 找 到 该 文件 中 的 如 下 指令 行 : 
AddType application/x-gzip .gz .tgz 
在 该 指令 后 加 入 如 下 指令 : 
AddType application/x-httpd-php .php .phtml 
重新 启动 Apache， 并 在 Apache 主 目录 下 建立 文件 test.php: 
<?php 
Phpinfo0:; 


在 浏览 器 中 输入 http://127.0.0.1/test.php， 按 Enter 键 ， 如 果 出 现 如 图 2.14 所 示 的 页 面 ， 则 PHP 安 
装 成 功 。 


人 说 明 : Apache2 默认 的 主 目 录 为 /usr/local/apache2/htdocs。 
技巧 : 安装 文件 的 路 径 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移 植 程序 时 带 来 的 


不 便 ， 选 择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹配 。 建 议 最 好 不 要 选择 
中 间 有 空格 的 目录 ， 如 E:\program Files\PHP， 这 样 做 会 导致 一 些 未 知 错误 或 户 溃 发 生 。 
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图 2.14 PHP 测试 


2.5 数据 库 设 计 


开发 一 个 功能 完善 的 网 上 社区 离 不 开 数 据 库 的 支持 ， 只 有 拥有 了 强大 的 数据 库 ， 网 上 社区 才能 够 
存储 大 量 的 数据 信息 ,实现 更 多 、 更 好 的 功能 来 吸引 更 多 的 社区 成 员 。 本 节 将 对 BCTY365 网 上 社区 数 
据 库 的 设计 进行 详细 介绍 。 


2.5.1 数据 库 分 析 


BCTY365 网 上 社区 是 一 个 中 型 的 面向 软件 开发 者 的 程序 ， 考 虑 到 开发 的 成 本 、 拱 配 的 合理 性 以 及 
操作 的 灵活 性 等 ， 使 用 了 MySQL 数据 库 。 从 成 本 考虑 MySQL 数据 库 是 完全 免费 的 ， 可 以 在 网 上 免费 
下 载 ; 从 匹配 的 角度 讲 PHP 与 MySQL 数据 库 一 直 是 公认 的 最 佳 搭档 ; MySQL 数据 库 不 但 可 以 在 命令 
模式 下 进行 操作 ， 而 且 还 配备 了 一 些 比较 流行 的 图 形 化 管理 工具 ， 如 phpMyAdmin 等 ， 可 以 轻松 地 对 
MySQL 数据 库 中 的 数据 进行 操作 。 


2.5.2 ”数据 库 概念 设计 


根据 上 述 各 节 对 BCTY365 网 上 社区 系统 做 的 需求 分 析 和 系统 设计 ， 规 划 出 BCTY365 网 上 社区 的 
实体 关系 E-R 图 。 其 中 包括 注册 用 户 实体 、 发 帖 信息 实体 、 回 帖 信息 实体 、 订 单 信息 实体 和 编程 词典 
信息 实体 ， 其 他 还 有 一 些 辅助 的 实体 ， 是 对 上 述 实体 中 内 容 的 补充 。 由 于 涉及 的 实体 较 多 ， 这 里 只 对 
注册 用 户 实体 、 发 帖 信息 实体 和 订单 信息 实体 的 E-R 图 进行 介绍 。 

1. 注册 用 户 实体 

注册 用 户 实体 用 于 存储 用 户 注册 信息 ， 包 括 编号 、 用 户 名 、 真 实 姓名 、 密 码 、 邮 箱 、 人 性别、 电话 、 


qq 
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QQ 号 码 、 家 庭 地 址 、 访 问 次 数 、 注 册 时 间 、 最 后 一 次 登录 时 间 、 卫 地 址 、 邮 政 编码 、 用 户 类 型 、 密 
码 提示 问题 、 密 码 答案 、 真 实 密码 、 表 情 图 、 发 帖 次 数 属性 。 注 册 用 户 实体 的 E-R 图 如 图 2.15 所 示 。 


E 


箱 


码 
Cssanm 7 ; 密码 提示 问题 


图 2.15 注册 用 户 实体 E-R 


2. 发 帖 信息 实体 


发 帖 信息 实体 用 于 存储 登录 本 社区 的 会 员 在 论坛 中 发 布 帖 子 的 相关 信息 ， 包 括 编号 、 用 户 名 ID、 
帖子 类 别 、 帖 子 标题 、 帖 子 内 容 、 发 帖 时 间 、 最 后 回复 时 间 、 表 情 图 、 访 问 次 数 、 是 否 项 帖 、 上 传 图 
片 属性 。 发 帖 信息 实体 的 E-R 图 如 图 2.16 所 示 。 


2.16 发 帖 信息 实体 的 E-R 图 


3. 订单 信息 实体 


订单 信息 实体 存储 用 户 在 线 购买 时 填写 的 订单 信息 ， 包 括 编号 、 用 户 名 、 性 别 、 家 庭 地 址 、 邮 政 
编码 、QQ 号 码 、 邮 箱 、 手 机 号 码 、 电 话 号 码 、 收 货 方式 、 邮 资 、 产 品 金额 、 订 单 时 间 、 订 单 号、 选择 
城市 等 属性 。 订 单 信息 实体 的 E-R 图 如 图 2.17 所 示 。 


2.5.3 创建 数据 库 及 数据 表 


在 BCTY365 网 上 社区 系统 中 应 用 的 是 db_bcty365 数据 库 ， 其 中 涉及 18 个 数据 表 ， 数 据 表 的 名 称 
和 功能 如 图 2.18 所 示 。 
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th_bb MylISAM ”gb2312_chinese_ci 。 编程 词典 版 本 信息 表 
th_bbqb MyISAM ”gb2312_chinese_ci 。 版 本 之 间 区 别 信息 表 
th_bbs MyISAM ”gb2312_chinese_ci 。 论坛 发 帖 信息 表 
tb_bccd MYISAM ”gb2312_chinese_ci 。 编程 词典 信息 表 
tb_city MylSAM ”gb2312_chinese_ci “城市 信息 表 
th_cjwt MyISAM ”gb2312_chinese_ci 。 常见 问题 信息 表 
tb_dd MyISAM gb2312_chinese_ci 。 订单 信息 表 
tb_reply MyISAM ”gb2312_chinese_cl 。 论坛 回帖 信息 表 
th_sjxz MylISAM ”gb2312_chinese_ci 。 软件 升级 下 载 信息 表 
tb_soft MyISAM gb2312_chinese_cl 。 软件 下 载 信息 表 
th_tell MyISAM ”gb2312_chinese_cl 。 社区 公告 信息 表 
th_type MyISAM ”gb2312_chinese_ci 。 社区 模块 类 型 信息 表 


th_type_big MylSAM ”gb2312_chinese_ci 。 论坛 大 类 信息 表 
tb_type_small MylISAM gb2312_chinese_ci 。 论坛 小 类 信息 表 
th_user MyISAM ”gb2312_chinese_ci 注册 用 户 信息 表 
th_xih MylSAM ”gb2312_chinese_ci 。 升级 下 载 序列 号 信息 表 
th_bccdl MyISAM ”gb2312_chinese_ci 。 编程 词典 简介 信息 表 
了 b_leaveword MylSAM gb2312_chinese_ci ”存储 客户 反馈 信息 


图 2.18 db_bcty365 数据 库 中 使 用 的 数据 表 
本 案例 中 创建 数据 库 和 数据 表 使 用 的 是 phpMyAdmin 图 形 化 管理 工具 ， 下 面 将 介绍 数据 库 和 数据 
表 的 创建 方法 ， 以 及 在 创建 过 程 中 需要 注意 的 一 些 问 题 。 
1. 数据 库 的 创建 
打开 phpMyAdmin 图 形 化 管理 工具 的 主页 ， 首 先 在 文本 框 中 输入 要 创建 的 数据 库 的 名 称 〈 如 db_ 


bcty365), 然后 在 下 拉 列 表 框 中 选择 要 使 用 的 字符 编码 格式 , 这 里 使 用 的 是 gb2312_chinese_ci, 如 图 2.19 
所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 数 据 库 创建 成 功 。 


总 技巧 : 创建 数据 库 的 过 程 中 ， 尽 量 使 用 与 程序 内 容 贴切 的 英文 字符 进行 命名 ， 有 助 于 对 数据 库 的 
理解 ; 如 果 使 用 AppServ 配置 PHP 开发 环境 , 那么 在 创建 数据 库 时 不 需要 指定 编码 的 格式 ， 
默认 值 为 gb2312_chinese_ci; 如 果 自 行 配置 开发 环境 ， 那 么 就 要 指定 编码 格式 为 gb2312_ 
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chinese_ ci， 和 否则 创建 数据 库 的 编码 格式 为 “latin1 swedish _ ci” ， 将 导致 数据 库 中 数据 出 现 
乱码 。 


localhost phpMyAdmin - 2.9.0.2 
回 服 务 器 版 本 : 5.0.24-comm 


L client version: 4.1.7 
» Protocol version| 10 输入 数据 库 名 
ps localho 守 via TCPAP 


PHP extensions: mysql 
全 Language 国 : [中文 -Chinese simpiiied 司 
人 重 phpMyAdmin 官方 网 站 

» [changeLog] [CVS] [usts] 
| 


Re 

晶 数 据 库 [ 
部 导出 
可 Import 


于 和 phpMyAdmin 


图 2.19 ” phpMyAdmin 管理 界面 
2. 创建 数据 表 


在 成 功 创建 数据 库 以 后 ， 接 下 来 就 是 创建 数据 表 ， 这 里 以 tb_bb 编程 词典 版 本 信息 表 为 例 ， 讲 解 


如 何 创建 数据 表 ， 以 及 在 创建 数据 表 的 过 程 中 都 需要 注意 哪些 问题 。 这 里 创建 一 个 名 字 为 tb_bb 的 数 
据 表 ， 包 括 3 个 字段 ， 如 图 2.20 所 示 。 


三 莉 在 数据 库 db_bcty365 中 创建 一 个 新 表 


名 jb bb ) Number offiel(s:[3 _) 
输入 新 表 名 称 输入 新 表 字段 个 数 于 用 可 


图 2.20 创建 tb_bb 数 据 表 


单 击 “ 执 行 ”按钮 后 ， 进 入 到 如 图 2.21 所 示 的 添加 字段 信息 的 页 面 中 ， 在 此 处 对 字段 进行 详细 的 
设置 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 、 主 键 和 索引 等 。 


胃 服务 器 : localhost ， 数据 库 : db_bcty365 ， 国 表 :tb_bb 
字段 类 型 @ 


长 度 慎 *1 Me —— 设 轩 主 刍 力 
-ee 10 not null 加 nt 本 gl 
bbname VARCHAR notnull 要 c 


[ereatetme DATETIME 一 not null 一 指定 编码 格式 


| 指定 数据 表 的 关 型 | Enh oom 0 


[ab2312 chinese ci 7 


图 2.21 添加 数据 表 中 字段 信息 


总 技巧 : 创建 数据 库 中 的 数据 表 时 ， 字 段 名 的 设计 尽量 要 与 数据 表 的 内 容 相符 合 ， 这 样 有 助 于 程序 
后 期 维护 和 修改 工作 的 进行 ， 能 够 直观 地 看 出 数据 表 的 作用 。 

如 果 使 用 AppServ 配置 PHP 开发 环境 ， 那 么 在 创建 数据 表 时 不 需要 指定 数据 表 类 型 和 编码 
格式 ; 如 果 使 用 自行 配置 的 PHP 开发 环境 , 那么 就 要 指定 数据 表 的 类 型 为 MyISAM 和 字符 
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的 编码 格式 为 gb2312_chinese_ ci， 否则 创建 的 数据 表 类 型 为 InnoDB， 而 编码 格式 为 
latinl swedish ci， 将 导致 该 数据 表 中 的 数据 复制 到 其 他 机 器 上 不 可 用 ， 并 且 数据 表 中 的 数 


据 出 现 乱码 。 


在 创建 数据 表 的 过 程 中 ， 一 定 要 为 数据 表 指 定 一 个 主键 ， 它 是 数据 表 的 唯一 标识 。 
掌握 数据 表 的 创建 方法 后 ， 就 可 以 自行 创建 数据 表 。 由 于 本 案例 中 涉及 的 数据 表 有 17 个 之 多 ， 这 


里 不 能 对 每 个 数据 表 的 功能 设计 进行 一 一 介绍 ， 所 以 只 给 出 几 个 重要 的 数据 表 的 设计 效果 图 供 广大 读 


者 参考 ， 其 他 数据 表 请 读者 参见 本 书 附带 的 光盘 。 数 据 表 的 设计 结构 如 图 2.22 一 图 2.24 所 示 。 


tb_user (注册 用 户 信息 表 ) 。 


图 服务 器 : localhost ， 忆 数据 库 : db_bcty365 ， 国 表 :tb_user 


注册 用 户 信 息 表 主 要 用 于 存储 本 社区 中 会 员 的 个 人 信息 。 该 数据 表 的 结构 如 图 2.22 所 示 。 


字段 大 型 整理 尾 性 Null 默认 镍 外 说 明 
a int(e) 否 auto_increment 自动 编号 ID 
Usernc varchar(50) 。 gb2312_chinese_cl 是 NULL 注册 用 户 各 
truename varchar(50) gb2312_chinese_ci 是 NULL 真实 姓名 
pwd varchar(50) gb2312_chinese_ci 是 WULL 注册 密码 
email varchar(50) 。 gb2312_chinese_cl 是 。 NULL 有 效 邮箱 地 址 
sex varchar(2) 。 gb2312_chinese_cl 是 NULL 性 别 
tel varchar(20) 。 gb2312_chinese_ci 是 WULL 联系 电话 
qq varchar(20) gb2312_chinese_ci 是 NULL 号 码 
address varchar(100) gb2312_chinese_cl 是 NULL 联系 地 址 
ogintimes int() 否 访问 次 数 
regtime datetime 否 注册 时 间 
lastlogintime datetme 否 最 后 一 次 登录 时 间 
ip varchar(20) gb2312_chinese_cl 否 Tf 地 址 
ww varchar(20) 。 gb2312_chinese_ci 是 NULL 邮政 编码 
usertype int(2) 否 用 户 类 型 
question varchart200) gb2312_chinese_cl 理 密码 提示 问题 
answer varchar(200) gb2312_chinese_cl 否 密码 提示 答案 
truepwd varchar(200) gb2312_chinese_ci 否 真实 密码 
photo varchar(50) gb2312_chinese_cl 否 表情 图 
pubtimes int(4) 是 0 发 帖 次 数 


图 2.22 注册 用 户 信息 表 结 构 
tb_ reply《〈 论 坛 回帖 信息 表 ) 。 


论坛 回帖 信息 表 主 要 用 于 存储 登录 会 员 在 本 社区 中 回复 帖子 的 信息 。 该 数据 表 的 结构 如 图 2.23 


所 示 。 
胃 服务 器 : localhost ， 轧 数据 库 : db_bcty365 ， 国 表 :tb_reply 


字段 类 型 整理 屋 性 Null 默认 额外 说 明 
过 int(8) 理 auto_increment 自动 编号 ID 
userid int(g) 否 0 注册 用 户 ID 
bbsid int(8) 要 0 发 布 帖子 表 I 
‘title varchar(200) gb2312_chinese_ci 是 NULL 回复 主题 
content mediumtext gb2312_chinese_ci 是 WUL 回复 内 容 
Createtime 。 datetime 是 NULL 回复 时 间 
mark nt 是 。 WULL 回复 记录 
photo varchar(80) 。 gb2312_chinese_ci 是 NULL 回复 图 片 


图 2.23 论坛 回帖 信息 表 结 构 
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回 tb bccd〈 编 程 词 典 信息 表 ) 。 
编程 词典 信息 表 主 要 用 于 存储 本 社区 的 在 线 订 购 模块 中 出 售 的 编程 词典 的 基本 信息 。 该 数据 表 的 
结构 如 图 2.24 所 示 。 


国 服务 器 : localhost ， 电 数据 库 : db_bcty365 ， 国 表 :tb_bccd 

字 及 类 型 整理 尾 性 Null 扶 认 额外 说 明 
int(8) auto_increment 自动 编号 ID 
varchar(200) gb2312_chinese_ci 篇 程 闻 典 名 称 
varchar(100) gb2312_chinese_ci 开发 者 
varchar(50) 。 gb2312_chinese_ci 
mediumtext 。 gb2312_chinese_ci 
mediumtext 。 gb2312_chinese_cl 
datetime 
varchar(100) gb2312_chinese_ci 
int8) 
fioat 


NULL 所 属 版 本 的 ID 
NULL 价格 


图 2.24 编程 词典 信息 表 结 构 


HE 
多 


2.6 公共 模块 设计 


2.6.1 数据库 连接 文件 


在 进行 程序 开发 的 过 程 中 ， 有 很 多 地 方 都 涉及 到 数据 库 的 应 用 ， 在 应 用 数据 库 之 前 首先 要 与 数据 
库 建立 连接 ， 因 此 可 以 将 数据 库 的 连接 代码 作为 一 个 公共 文件 进行 存储 ， 在 需要 使 用 数据 库 连 接 文 件 
的 地 方 直接 调用 该 文件 即 可 ， 无 须 重复 编写 相同 的 代码 ， 既 减少 了 代码 的 元 余 ， 也 便于 对 数据 库 连 接 
文件 进行 修改 。 在 本 项 目 中 笔者 将 数据 库 的 连接 代码 存储 于 conn.php 中 。conn.php 文件 的 代码 如 下 : 

例 程 01 代码 位 置 : 光盘 \TM\02\bcty365\conn\conn.php 


<?php 

0 S$conn=mysql_ connect("localhost"."root"."root): ”// 连 接 数 据 库 服务 器 

@ mysql_select_db("db_bcty365".Sconn): // 连 接 指定 的 数据 库 

目 mysql_query("set names gb2312"): /对 数据 库 中 的 编码 格式 进行 转换 ， 避 免 出 现 中 文 乱码 的 问题 
?> 

Ah 代码 贴 二 


@ mysql conect0: 连接 MySQL 服务 器 ， 服 务 器 的 用 户 名 为 root， 密 码 为 Toot。 
@ mysql select db0: 用 于 连接 指定 的 MySQL 数据库， 数据 库 名 为 db_bcty365。 
上 @ set names gb2312: 指定 数据 库 中 字符 的 编码 格式 为 gb2312. 


成 功 创建 conn.php 文件 后 , 在 需要 进行 数据 库 操作 的 程序 中 , 通过 include 或 者 其 他 包含 语句 调用 
conn.php 文件 即 可 , 无 须 再 编写 连接 数据 库 的 程序 代码 。 应 用 include 语句 包含 conn.php 文件 的 代码 如 下 : 
<?php 


include ("conn/conn.php"): // 包 含 数据 库 文件 
>. 


第 2 章 BCTY365 网 上 社区 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


2.6.2 ”将 文本 中 的 字符 转换 为 HTML 标识 符 


在 输出 数据 库 中 数据 的 过 程 中 ， 有 必要 将 数据 中 的 一 些 特殊 字符 转换 为 HTML 标识 符 ， 这 样 可 以 
避免 一 些 不 必要 的 麻烦 。 例 如 ， 在 输出 一 个 程序 的 执行 代码 的 过 程 中 ， 如 果 不 对 其 进行 转换 ， 那 么 输 
出 的 将 不 是 程序 的 代码 ， 而 是 程序 的 执行 结果 。 这 里 将 文本 中 字符 的 转换 编写 到 一 个 自 定义 函数 
unhtml0 中 ， 保 存 到 function.php 文件 中 ， 将 其 作为 一 个 公共 模块 来 使 用 ， 当 需要 使 用 时 直接 调用 
function.php 文件 即 可 。function.php 文件 包含 两 个 自 定义 函数 ， 即 unhtml0 和 msubstr0 函 数 ，unhtml() 
函数 用 于 将 数据 中 的 特殊 字符 转换 为 HTML 标识 符 ; msubstr0 函 数 用 于 对 字符 串 进 行 指定 长 度 的 截取 。 
其 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 \TM\02\bcty365\functionphp 


<?php 
function unhtml($content) { // 定 义 自 定义 函数 的 名 称 
0 Scontent=htmlspecialchars($content); // 转 换文 本 中 的 特殊 字符 
@ Scontent=str_replace(chr(13)."<br>",Scontent): // 蔡 换文 本 中 的 换行 符 
S$content=str_Teplace(chr(32),"&nbsp:",$contenb: // 蔡 换文 本 中 的 &nbsp: 
S$content=str_ replace("[_["."<".Scontent]: // 蔡 换文 本 中 的 大 于 号 
Scontent=str_replace(") )",">",.$content): // 蔡 换文 本 中 的 小 于 号 
Scontent=str replace("| |"," ",$content): // 蔡 换文 本 中 的 空格 
目 return trim($content); // 删 除 文本 中 首尾 的 空格 


} 
/定义 一 个 用 于 截取 一 段 字符 串 的 函数 msubstrO 
function msubstr($str,Sstart,Slen){ 。“”//$str 指 的 是 字符 串 ，S$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 


$strlen=$start+ $len: /用 $strlen 存 储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
for($i=0:$i<$strlen.Si+H){ /通过 for 循 环 语句 ， 循 环 读 取 字 符 串 
@ if(ord(substr($str,$i.1))>0xa0){ /如 果 字 符 串 中 首 个 字 节 的 ASCII 序 数值 大 于 0xa0， 则 表示 为 汉字 
Stmpstr.=substr($srt,$i,2): // 每 次 取出 两 位 字符 赋 给 变量 Stmpstr， 即 等 于 一 个 汉字 
Si++; /变量 自 加 1 
jelse{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 Stmpstr 
S$tmpstr.=substr($str.$i.1):} 
} 
Tetum $tmpstr; /输出 字符 串 
}?> 
< 代码 贴 十 


@ htmlspecialchars0: 将 特殊 字符 转换 成 HTML 格式 ， 而 不 会 将 所 有 字符 都 转换 成 HTML 格式 。 

@ str replace 0: 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ， 参 数 &count 表示 替换 字符 串 执行 的 次 数 。 
四 trim0: 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 。 

@ substr0: 从 指定 的 字符 事 中 按照 指定 的 位 置 截取 一 定 长 度 的 字符 .。 
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2.7 前 台 首页 设计 


当今 时 代 ， 很 多 人 都 十 分 重视 对 事物 的 第 一 印象 ， 第 一 印象 基本 上 就 决定 对 某 个 事物 的 看 法 和 态 
度 ， 在 网 络 中 更 是 如 此 ， 网 站 给 人 的 第 一 印象 如 果 不 好 ， 那 么 就 会 有 很 多 人 因此 而 不 去 浏览 该 网 站 ， 
无 论 网 站 的 内 容 是 否 丰富 。 可 以 说 网 站 首页 设计 的 成 功 与 否 直接 影响 着 整个 网 站 的 发 展 。 


2.7.1 ”前台 首页 概述 


网 站 首页 是 整个 网 站 的 脸面 ， 既 要 突出 企业 的 形象 ， 又 要 展示 出 网 站 强大 的 功能 。 如 果 网 站 首页 
设计 得 非常 成 功 ， 那 无 疑 是 为 整个 网 站 的 成 功 增添 了 一 个 夸 码 。BCTY365 网 上 社区 首页 的 设计 以 企业 
的 品牌 形象 为 基础 ， 全 力 打 造 网 站 的 整体 功能 ， 重 点 推出 企业 的 软件 产品 ， 具 体内 容 如 下 : 

回 ”网 站 菜单 导航 : 包括 首页 、 技 术 支 持 、 在 线 订 购 、 社 区 论坛 、 软 件 下 载 、 升 级 下 载 、 购 买 须 
知 、 联 系 我 们 。 

用 户 注册 和 登录 模块 :实现 用 户 注册 、 会 员 登 录 、 找 回 密码 和 修改 密码 的 功能 。 
网 站 公告 : 主要 用 于 发 布 社区 中 的 一 些 新 消息 和 重大 事件 。 

编程 词典 模块 : 推广 企业 的 软件 产品 。 

软件 下 载 模块 : 展示 企业 提供 的 适用 版 和 免费 的 软件 产品 。 

常见 问题 模块 : 列举 出 编程 中 常见 问题 的 解决 方案 。 

社区 论坛 模块 : 浏览 社区 论坛 中 的 部 分 帖子 。 

升级 下 载 模块 : 提供 一 些 软件 的 升级 版 本 下 载 。 

上 述 内 容 就 是 BCTY365 网 上 社区 首页 中 体现 出 的 内 容 ， 为 了 更 加 直观 地 了 解 网 上 社区 首页 的 设计 ， 
这 里 先 预 览 一 下 社区 首页 , 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\02\bcty365\index.php， 如 图 2.25 所 示 。 

BCTY365 网 上 社区 首页 的 设计 看 上 去 很 复杂 ， 有 很 多 个 版 块 组 成 ， 但 实现 的 过 程 非常 简单 。 总 体 
架构 使 用 一 个 2 行 3 列表 格 和 一 个 3 行 3 列 的 表格 ， 将 其 分 割 成 不 同 的 版 块 ， 然 后 使 用 脚本 语句 从 数 
据 库 中 读 取 数据 ， 最 后 将 数据 循环 输出 到 页 面 中 ， 其 中 网 站 的 头 尾 文件 使 用 include 包含 语句 调用 。 首 
页 的 框架 结构 如 图 2.26 所 示 。 


办 办 罗 办 欠 欠 罗 


2.7.2 ”前台 首页 技术 分 析 


作为 网 站 首页 ， 不 一 定 要 具有 什么 特殊 的 技术 或 者 功能 ， 应 该 是 以 简捷 、 鲜 明 ， 突 出 企业 形象 ， 
展示 网 站 的 功能 为 主 。 即 使 使 用 的 是 一 个 静态 页 面 ， 只 要 能 够 将 内 容 表达 人 全面、 完整， 那么 这 个 首页 
设计 也 是 非常 成 功 的 。 

在 本 案例 首页 的 设计 中 ， 应 用 到 一 个 文字 循环 滚动 的 技术 ， 通 过 该 技术 来 输出 社区 中 发 布 的 公告 
信息 。 该 技术 是 通过 JavaScript 脚本 和 div 标签 来 共同 实现 的 ， 其 实现 的 原理 是 : 首先 创建 一 个 div 标 
签 ， 然 后 在 div 标签 中 输出 公告 信息 ， 最 后 通过 JavaScript 来 对 div 标签 进行 操作 ， 实 现 div 标签 中 内 
容 的 滚动 输出 。 该 技术 的 实现 在 index.php 页 中 完成 ， 其 中 使 用 的 javascript 脚本 的 代码 如 下 : 
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, | yg ee ， ， 过 对 生 对 划 中 的 发 性 方法 进行 引用 的 ， 关 


Digert 机 sorithm 5 在 9 年代 初 | LBAANY MANAGE 氏 用 中 还 ， 一 舱 久 况 下 ， 粮 斥 是 通过 对 旬 对 
出 C007/00719) 只 是 具备 系 项 功能 的 抽象 模型 ,实际 太 用 中 


Taom-ll-06 1 


程序 出 试 [2007411721] 
程序 而 弃 [2o07/11/21] 


图 2.25 BCTY365 网 上 社区 首页 


图 2.26 网 站 首页 的 框架 结构 
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例 程 03 ”代码 位 置 : 光盘 \TM\02\bcty365\index.php 
<script language="JavaScript"> 


marqueesHeight=222:; // 定 义 输出 标签 的 高 度 
stopscroll=false; // 定 义 stopscroll 的 默认 值 为 false 
with(marquees){ // 编 辑 marquees 标 签 的 属性 
style.width=0; // 定 义 初始 宽 为 0 
style.height=marqueesHeight: // 定 义 marquees 标 签 的 高 为 222 
style.overflowX="visible"; // 定 义 值 为 显示 
style.overflowY="hidden": /定义 值 为 隐藏 
noWrap=true; 
‘onmouseover=new Function("stopscroll=true"):; // 当 鼠标 经 过 时 执行 stopscroll=true 
‘onmouseout=new Function("stopscroll=false"): // 当 鼠标 离开 时 执行 stopscroll=false 


) 

// 创 建 一 个 新 的 div"templayer" 与 div"marquees" 进 行 连接 ， 实 现 不 间断 的 循环 输出 内 容 
document.write('<div id="templayer" style="position:absolute;z-index: 1:visibility:hidden"></div>"); 
preTop=0; currentTop=0; 


function initO{ 
templayer.innerHTML=""; /设置 templayer 的 初始 值 为 空 
while(templayer.offsetHeight<marqueesHeight) { // 判 断 当 templayer 的 高 度 小 于 marquees 的 高 度 时 


templayer.innerHTML+=marquees.innerHTML: // 将 templayer 的 值 赋 给 marquees 


} 
marquees.innerHTML=templayer.innerHTML+templayer.innerHTML: // 将 templayer 的 值 累加 


setInterval("scrollupO".50): // 间 隔 50 毫 秒 执行 一 次 scrollup0 函 数 
) 
function scrollupO{ /实现 滚动 输出 
这 stopscroll 一 tmue) return: /判断 如 果 stopscroll 一 tue， 不 执行 循环 
preTop=marquees.ScrollTop: 
Inarquees.ScTOIITop+=1; 
这 preTop 一 marquees.scrollTop){ 


Imarquees.ScrollTop=templayer.offsetHeight-marqueesHeight: 
marquees.scrollTop+=1: 


) 


</script> 


在 div 标签 中 ， 主 要 是 输出 数据 库 中 存储 的 公告 信息 ， 并 且 对 输出 的 信息 进行 截取 和 替换 ， 规 范 
输出 的 内 容 。div 标签 中 的 程序 代码 如 下 : 


<div id=marquees > ”<!--> 创 建 一 个 div 标 签 <!--> 
<table width="200" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<2php /从 数据 库 中 读 取 公告 数据 
$sql=mysql query("select id.title.createtime from tb _tell order by createtime desc limit 0.10".$conm): 
Sinfo=mysql fetch_array($sq]): 
这 $info 一 包 lse){ // 判 断 当 $info=false 时 执行 下 面 的 内 容 
?> 
<t> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</t> 
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<?php jelse{ 
SEl; // 定 义 变量 $i=1 
do{ // 执 行 do...while 循 环 语句 
?> 
<t> 
<td height="25" style="padding:4"> 
<a href="tellinfo.php?id=<?php echo $infofid]:?>" class="al"> 
<2php 
这 Si 一 1){ // 判 断 当 $i==1 时 ,将 输出 的 内 容 设置 为 红色 
echo "<font color=red>"; 
} 
echo $i.".&nbsp:;"; 
echo unhtml(msubstr($infoftitle],0,50)): // 应 用 自 定义 函数 对 输出 的 内 容 进行 控制 
ifstrlen($info[title])>50){ // 当 输出 内 容 的 长 度 超过 50 个 字符 时 用 "…" 代 替 
Boho. 
} 
echo "(".str_replace("-","/",Sinfo[createtime]).")"; // 将 输出 的 公告 时 间 中 的 "-" 用 "" 替 代 
这 $i 一 1){ echo "</font>": } 
> </a> 
</td> 
</t> 
<?php 
Si++; 
}while($info=mysql _fetch_array($sqD): /do…while 循 环 语句 结束 
?> 
</table> 
</div> 


在 首页 中 使 用 滚动 条 是 一 个 比较 不 错 的 方法 ， 通 过 其 可 以 增加 网 页 的 动态 效果 ， 增 加 网 页 的 观赏 
性 ， 而 且 不 会 影响 到 网 页 的 浏览 速度 。 


2.7.3 前 台 首 页 的 实现 过 程 


国 前 台 首页 使 用 的 数据 表 : tb tell、tb_soft、tb bccd、tb bb、tb_cjwt、tb_bbs、tb_sjxz 

开发 网 站 首页 主要 就 是 连接 数据 库 ， 从 数据 库 中 读 取 数 据 ， 最 后 应 用 循环 语句 将 数据 库 中 数据 输 
出 到 前 台 页 面 。 由 于 使 用 的 代码 较 多 ， 而 且 多 数 都 是 重复 使 用 ， 所 以 这 里 只 给 出 首页 中 公告 发 布 模块 
的 代码 。 

公告 发 布 模块 主要 实现 从 数据 库 中 读 取 公 告 数据 ， 将 数据 在 首页 中 滚动 输出 ， 并 且 对 公告 信息 的 
长 度 进行 控制 , 保证 内 容 的 整齐 、 规范 。 具体 详细 代码 可 以 参考 本 书 光盘 中 光盘 \TM\02\bcty365\index.php 
文件 ，index.php 文件 的 部 分 代码 如 下 : 

例 程 04 ”代码 位 置 : 光盘 \TM\02\bcty365\index.php 

<?php include_once("top.php"): /获取 头 部 文件 ?> 

…// 省 略 部 分 代码 

<?php 
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// 读 取 数 据 库 中 公告 表 中 的 数据 
S$sql=mysql query("select id.title.createtime from tb _tell order by createtime desc limit 0.10",$conn): 
Sinfo=mysql fetch array($sq)): // 执 行 读 取 数据 表 中 数据 的 语句 
这 S$info 一 false){ // 如 果 返 回 值 为 空 则 执行 下 面 的 语句 
?> 
<tr> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</t> 
<2php 
jelse{ /如 果 返 回 值 不 为 空 则 执行 下 面 的 do.…while 循 环 语句 
$i=1; 
do{ 
ve 
<h> 
<td height="25" style="padding:4"><a href="tellinfo.php?id=<?php echo $info[id]:?>" class="al"> 
<?php 
TD /判断 当 变量 $i=1 时 ， 输 出 的 内 容 以 红色 字体 显示 
echo "<font color=red>"; 
echo $i.".&nbsp:"; 
© echo unhtml(msubstr($info[title].0.50)): 
[2 这 strlen($info[title])>50){ // 如 果 标 题 长 度 大 于 50 个 字符 ， 则 以 省 略 号 代替 
RGB 
y 
目 echo "(".str_replace("-","/",Sinfo[createtime]).")": /输出 公告 发 布 的 时 间 ， 并 且 将 其 中 的 "使 用 "-" 蔡 换 
if($i==1){ 
echo "</font>"; 
} 
?> 
</a> 
</td> 
</t> 
<?php 
SitH+: // 变 量 自 加 1 
}while($info=mysql fetch array($sql)): //do…while 循 环 语句 结束 
} 
?> 
区 // 省 略 了 部 分 代码 
<?php 
include_once("bottom.php"): // 调 用 网 站 的 尾 文件 
?> 
人 代码 贴 十 


@ 应 用 unhtml0 和 msubstr0 自 定义 函数 去 除 输出 字符 串 中 的 空格 和 控制 输出 字符 串 的 长 度 。 
@ strlen0: 获取 指定 字符 串 的 长 度 。 
@ str replace0: 实现 字符 囊 的 替换 . 
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2.8 注册 模块 设计 


2.8.1 注册 模块 概述 


BCTY365 网 上 社区 系统 为 了 更 好 地 与 广大 网 民 朋友 进行 交流 和 沟通 ， 创 建 了 一 个 会 员 注 册 模 块 。 
通过 该 模块 ， 系 统 可 以 有 效 地 对 用 户 信 息 进行 采集 ， 并 将 合法 的 用 户 信息 保 存 到 指定 的 数据 表 中 ， 实 
现 与 用 户 的 长 期 沟通 和 交流 。 既 然 设 置 了 会 员 注 册 模块 ， 那 么 在 系统 中 就 要 为 会 员 提供 一 些 特殊 的 权 
限 。 在 本 系统 中 注册 会 员 可 以 拥有 如 下 权限 : 在 本 社区 的 论坛 中 发 布 和 回复 帖子 ， 在 技术 支持 模块 中 
发 表 留言 ， 在 升级 下 载 模块 中 下 载 软件 升级 包 等 ， 而 且 可 以 进行 修改 密码 和 找 回 密码 。 用 户 注 册 模块 
的 运行 结果 如 图 2.27 所 示 。 


您 已经 成 Th 的 中 请 了 本 站 的 用 尸 各 : DC 程序 内 加 果 坟 出 为 本 站 的 用 户 ， 您 还 慨 局 塘 写 以 下 会 各 
广 意 : 以 下 注册 信息 灼 为 必 尖 内 容 ? 


性别: 大 司 

部 条 地址 :PSsine em， [内 了 便于 工作 人 员 与 人 联系 ,请 填写 正确 的 -si 地址 + ) 
WR: [aaa 

部 本 [I 成， 并 且 为 8(Y1) 

Wy: [el 


潜 阿 住址 : 
头 佛 志 择 4eif 忆 稚 
| 


区 的 答案 


ai: Fr 5861 
ZE | ES | 
图 2.27 用 户 注册 模块 的 运行 结果 


的 了 能 凶 找 同 天 夫 的 区 码 ,请 记 住 该 短 要? ) 


2.8.2 ”注册 模块 技术 分 析 


在 会 员 注册 模块 中 ， 必 不 可 少 的 就 是 要 对 用 户 输入 的 信息 进行 判断 ， 首 先 判断 用 户 填 写 的 注册 信 
息 中 哪些 是 必须 填写 的 ， 哪 些 可 以 不 填写 ， 然 后 进一步 判断 输入 的 信息 是 否 合理 合法 ， 例 如 ， 判 断 输 
入 的 邮编 的 格式 是 否 正确 ， 判 断 输 入 邮箱 的 格式 是 否 正确 等 。 对 表单 中 提交 数据 进行 判断 最 常用 的 办 
法 就 是 使 用 JavaScript 脚本 ， 也 可 以 使 用 正则 表达 式 。 下 面 讲解 在 本 模块 中 是 如 何 通过 JavaScript 实现 
表单 提交 数据 验证 。 

操作 原理 是 : 在 form 表单 中 调用 onsubmit 事件 ， 通 过 该 事件 调用 指定 的 JavaScript 脚本 ， 执 行 
chkinputO 自 定义 函数 ， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 实 现 对 表单 中 提交 数据 
进行 判断 ， 判 断 输入 的 内 容 是 否 为 室 ， 判 断 内 容 的 格式 是 否 正确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 


q 
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提示 对 话 框 ， 并 将 鼠标 的 焦点 指定 到 出 错 的 位 置 。 具 体 的 JavaScript 脚本 代码 如 下 : 
<script language="JavaScript" type="text/javascript"> 
function chkinput(form){ // 定 义 一 个 函数 

if(form .tel.value=—""){ // 判 断 tel 文 本 框 中 的 值 是 否 为 空 
alert(" 请 填写 联系 电话 !"): // 如 果 为 空 则 输出 "请 填写 联系 电话 !" 
form.tel.selectO|: // 返 回 到 tel 文 本 框 
Tetum(false): 

if(form.email.value—""){ // 浏 断 email 文 本 框 的 值 是 否 为 空 
alert(" 请 输入 E-mail 地 址 1"); // 如 果 为 空 则 输出 "请 输入 E-mail 地 址 !" 
form.email.select(): // 返 回 到 email 文 本 框 
retum(false); 


4 
Var i=form.email.value.indexOf("(@"): 
Var j=form.email.value.indexOf("."); 
// 进 一 步 判 断 邮箱 的 格式 是 否 正确 ,是 否 包含 "@" 和 "." 
这 (Gi<O)llG-j>O)lG<O)){ 
alert(" 请 输入 正确 的 E-mail 地 址 !"); 


form.email.select(): // 返 回 到 email 文 本 框 
return(false); 
} 
return(true): // 提 交 表单 
y 
</script> 


上 述 代 码 中 ， 只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进行 判断 时 ， 只 是 判断 其 是 
否 为 空 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正 
确 可 以 采用 下 面 的 方法 : 通过 正则 表达 式 的 preg_match0 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进行 
判断 。 

preg_match() 函 数 的 语法 格式 如 下 : 

int preg_match ( string pattern, string subject [, array matches [, int flags]] ) 


preg_match() 函 数 的 参数 说 明 如 表 2.3 所 示 。 


表 2.3 ”preg_match() 函 数 的 参数 说 明 
参数 说 明 
Pattern 必要 参数 。 需 要 匹配 的 正则 表达 式 


Subject 必要 参数 。 输 入 的 字符 串 


可 选 参 数 。 输 出 的 搜索 结果 的 数组 ， 如 $out[0] 将 包含 与 整个 模式 匹配 的 结果 ，Sout[1] 将 包含 与 第 一 个 捕 


Matches | 获 的 括号 中 的 子 模式 所 匹配 的 结果 ， 依 次 类 扒 


可 选 参数 。 标 记 : PREG_OFFSET_CAPTURE， 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 
量 ， 本 标记 自 php 4.3.0 起 可 用 


Flags 


通过 preg_match0 函 数 判 断 电话 号 码 的 格式 是 否 正 确 的 方法 如 下 : 首先 定义 一 个 用 于 判断 电话 号 码 
格式 的 正则 表达 式 ， 代 码 如 下 : 
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/0d{3}-)0d{8)sI0d{4} 047 0d{4 -DCdft8D3/ 
正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 的 限制 ， 对 区 号 从 字符 串 的 开 
始 进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “0” 中 的 内 容 作 为 一 个 原子 使 用 ; 使 
用 “\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ; 使 用 “{}” 来 对 前 字符 进行 重 
复 匹配 ;使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 
然后 将 该 正则 表达 式 应 用 到 preg_match0 函 数 中 ,对 表单 提交 的 电话 号 码 进 行 判 断 ， 如 果 正 确 则 继 
续 执行 ， 否 则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 ， 代 码 如 下 : 


<?php 
Stel="0431-84978981"; // 定 义 一 个 电话 号 码 的 变量 
if(preg_match("/^(\d{3}-)Qd{8})SI"Qd{4}-)Qd{7))SI"Qd{4}- ea ,StelScounts)){ 
echo "您 输入 的 电话 号 码 格式 正确 1"; 出 字符 串 
}else{ 
echo "<script>alert( 您 输入 的 电话 号 码 的 格式 不 正确 !1"):history.backQ0</script>"; 
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?> 
2.8.3 ”注册 模块 的 实现 过 程 


国 ”注册 模块 使 用 的 数据 表 : tb_user 

注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅 读 注册 服务 条 款 ， 然 后 填写 用 户 注册 的 用 户 名 和 密码 ， 提 
交 后 由 系统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 填 写 详细 的 注册 信息 ， 将 
数据 提交 到 表单 处 理 页 进行 处 理 ， 最 后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 用 户 注册 模块 的 实 
现 过 程 主要 由 3 个 文件 完成 :register.php 用 于 输出 注册 服务 条 款 ， 以 及 填写 注册 的 用 户 名 和 密码 ， 并 
且 判 断 注册 的 用 户 名 和 密码 是 否 被 占用 ;getuserinfo .php 文件 用 于 填写 详细 的 注册 信息 ， 并 且 在 表单 中 
应 用 数字 验证 码 技术 ; savereginfo .php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 , 将 数据 保存 到 指定 的 数 
据 表 中 。 

在 savereginfo.php 文件 中 ， 首 先 连 接 数 据 库 ， 然 后 获取 到 表单 中 提交 的 数据 ， 并 且 判 断 提交 的 用 
户 名 是 否 被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代码 如 下 : 


例 程 05 ”代码 位 置 : 光盘 \VTMVWO2\bcty36S\saverreginfo php 


<?php session_start(): /初始 化 session 变 量 
include_once("conn/conn.php"): // 连 接 数 据 库 

Susemc=trim($_ POST[usernc]): // 获 取 注 册 的 用 户 名 

/判断 指定 的 用 户 名 是 否 存在 

$sql=mysql_query("select usernc from tb_user where usernc=". $usernc."".$conn); 

$info=mysql fetch_array($sq]): // 按 指定 条 件 检索 数据 信息 
if($info!=false){ /如 果 查 询 结果 不 为 室 ， 则 执行 以 下 操作 


echo "<script language='javascript>alert( 对 不 起 ， 该 昵称 已 被 其 他 用 户 使 用 !):history.backO:</script>": 
exit: 


} 


Sxym=trim($ POST[xym]): // 去 除 变 量 两 边 的 空格 
$num=$ POST[num]:; // 接 收 变量 值 
if(strval($xym)!=strval($num)){ 
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echo "<script>alert( 验 证 码 输 入 错误 !):window-.location hre 人 registerphp':</script>": 


exit:} 
// 对 表单 提交 的 数据 进行 处 理 
Struepwd=trim($ POST[pwd1]): // 获 取 真 实 密码 
Spwd=md5($truepwd): // 获 取 加 密 密 码 
Struename=trim($ POST[truename]): // 获 取 真实 姓名 
$email=trim($_ POST[email]): // 获 取 邮 箱 地 址 
$sex=$ POST[sex]: // 获 取 性 别 
Stel=trim($_POST[tel]): // 获 取 电 话 
$yb=trim($ POST[yb]): /获取 邮政 编码 
$qq=trim($_ POST[qq]): /获取 QQ 
$address=trim($_ POST[address]): /获取 地 址 
$question=trimn($_POST[question]): /获取 提示 问题 
S$answer=trim($_ POST[answer]): /获取 问题 答案 
$ip=getenv("REMOTE ADDR"): /获取 客户 端的 
$logintimes=1: /指定 访问 次 数 
Sregtime=date("Y-m-j H:i:s"); /获取 时 间 
Slastlogintime=$regtime; 
Susertype=0; // 指 定 用 户 类 型 ,默认 为 0 
Sphoto=$_POST["photo"]: // 获 取 头 像 
// 将 表单 中 提交 的 数据 存储 到 数据 表 中 


if(mysql_query("insert into 

tb_user(usernc,truename,pwd,email,sex,tel,qq,address,logintimes,regtime,lastlogintime,ip,usertype,yb,question,answer,truepwd, 
photo) 
values('$usernc','$truename','$pwd','$Semail','$sex','$tel','$qq','$address','$logintimes','Sregtime','$lastlogintime','$ip','$usertype','$ 
yb','$question','$answer','$truepwd','Sphoto")", Sconn)){ 

session register("unc"); 

$_SESSION["unc"]=$useme: 

echo "<script>alert(' 注 册 成 功 !"):window.location.href='index.php':</script>": 


jelse{ /如 果 添 加 操作 失败 ， 则 给 出 提示 
echo "<script language='javascript>alert( 对 不 起 ,注册 失败 !"):history.backQ:</script>"; 
exit; // 退 出 

} 

> 


2.9 技术 支持 模块 设计 
技术 支持 模块 主要 是 从 浏览 者 的 角度 进行 设计 ， 存 储 大 量 技术 问题 的 解决 方案 数据 ， 为 浏览 者 查 
阅 提 供 方便 ， 而 且 设 计 一 个 企业 与 客户 沟通 的 平台 ， 能 够 随时 了 解 客 户 或 者 会 员 的 意见 和 需求 。 
2.9.1 技术 支持 模块 概述 
技术 支持 模块 主要 由 3 个 子 模块 组 成 ， 包 括 常见 问题 、 客 户 反馈 和 联系 方式 。 常 见 问题 模块 主要 


用 于 展示 编程 中 一 些 常 见 问题 的 解决 方案 或 者 方法 ， 为 浏览 者 解决 编程 中 的 疑难 问题 提供 方便 ;客户 
反馈 模块 主要 用 于 收集 和 获取 来 自 客户 的 需求 和 意见 ;联系 方式 模块 主要 用 于 展示 企业 的 形象 和 具体 
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的 联系 方式 。 
2.9.2 ”技术 支持 模块 技术 分 析 


技术 支持 模块 中 在 对 常见 问题 解决 方案 的 数据 进行 输出 时 ， 使 用 了 分 页 处 理 技 术 ， 该 技术 的 设计 
思路 是 : 从 数据 库 中 读 取 数据 ， 获 取 数据 总 量 ， 在 每 页 中 显示 20 条 数据 ， 根 据 数据 总 量 和 每 页 显示 的 
条 数 对 数据 进行 分 页 处 理 ， 计 算出 有 多 少 页 和 当前 显示 的 页 码 ， 实 现 首页 、 上 一 页 、 下 一 页 和 尾 页 之 
间 的 页 面 跳 转 。 具体 的 设计 思路 可 以 参考 cjwt.php 文件 中 的 代码 注释 和 代码 贴 士 。cjwt.php 文件 的 程序 
代码 如 下 : 

例 程 06 ”代码 位 置 : 光盘 \TM\02\bcty365\cjwt.php 


<?php 
S$sql=mysql_query("select count(*) as total from tb_cjwt".Sconn): // 读 取 数 据 库 中 数据 
Sinfo=mysql_fetch_array($sq)); // 返 回 数据 
© Stotal=S$info[total]; 
// 判 断 字段 total 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
这 Stotal 一 0){ 
?> 
<t> 
<td height="22" colspan="2"><div align="center"> 对 不 起 ， 暂 无 常见 问题 !</div></td> 
</t> 
<?php 
jelse{ /如 果 不 为 空 ， 则 执行 下 面 的 内 容 
@ if(!isset($_GET["page"]) || !is_ numeric($_GET["page"]){ // 判 断 $_GET 获 取 的 page 的 值 是 否 存在 
@ Spage=1: /如果 不 存在 ， 则 设置 变量 的 值 为 1 
jelse{ 
@ $page=intval($_GET["page"]): /如 果 存 在 ， 则 获取 变量 $_GET 的 值 
} 
// 设 置 变量 Spagesize, 每 页 显示 的 数据 数据 量 为 20 
© Spagesize=20: 
if($total%$pagesize—0){ // 如 果 变 量 的 值 为 0 
© Spagecount=intval($total/$pagesize): // 获 取 变量 的 整数 值 
Yelsef 
[7 Spagecount=ceil($total/$pagesize); // 如 果 不 为 0 则 获取 实际 的 整数 值 
} 
// 读 取 数 据 库 中 数据 ， 按 照 时 间 进 行 降 索 排列 
S$sql=mysql query("select * 位 om tb cjwt order by createtime desc limit ".($page-1)*$pagesize.",$pagesize 
",$conn); 
while($info=mysql fetch array($sql){ 
?> 
人 /省 略 部 分 代码 
<table width="600" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<t> 


<td width="479"><div align="left">&nbsp:&nbsp: 共 有 常见 问题 &nbsp:<?php echo $total:?>&nbsp: 条 &nbsp: 
每 页 显示 &nbsp:<?php echo $pagesize:?>&nbsp: 条 &nbsp: 第 &nbsp:<?php echo $page:?>&nbsp: 页 / 共 &nbsp:<?php echo 
$pagecount?>&nbsp: 页 </div></td> 

<td width="269"><div align—"right"> 
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e@ <a href="<?php echo $ SERVER["PHP SELF"]?>?page=1" class="al"> 首 页 </a>&nbsp: 
<a hre 人 "<?php echo $ SERVER["PHP SELF"]?>?page=<?php 
if($page>1) // 判 断 如 果 页 码 大 于 1 
echo $page-1: // 输 出 前 一 页 
else 
echo 1: 
?>" class="al"> 上 一 页 </a>&nbsp: 
<a href="<?php echo $_ SERVER["PHP SELF"]?>?page=<?php 
这 $page<$pagecount) /如 果 页 码 小 于 总 页 数 
echo $pagetl; 
else 
echo $pagecount: // 输 出 下 一 页 
?>" class="al"> 下 一 页 </a>&nbsp: 
<a href="<?php echo $_ SERVER["PHP_ SELF"]?>?page=<?php echo $pagecount:?>”class="al"> 尾 页 
</a>&nbsp:&nbsp:</div> 
</td> 
< 代码 贴 十 


@ Sitotal: 为 数据 库 中 数据 总 的 记录 数 。 

四 isset(0: 检测 变量 是 否 已 经 设置 ; is_numeric() 检 测 变量 是 否 为 数字 或 者 数字 字符 囊 。 
四 Spage: 变量 为 页 码 中 的 第 几 页 。 

@ intval0: 获取 变量 的 整数 值 。 

@ $pagesize: 表示 在 每 页 中 显示 多 少 条 数据 。 

@ $pagecount: 表示 所 有 的 数据 可 以 分 成 多 少 页 。 

@ ceil0: 获取 变量 中 的 整数 值 ， 这 里 用 于 获取 页 码 的 整数 值 。 

@ $_SERVER["PHP SELF"]: 服务 器 变量 ， 这 里 用 于 获取 网 页 的 链接 地 址 。 


2.9.3 ”常见 问题 的 实现 过 程 


国 ”常见 问题 使 用 的 数据 表 : tb_cjwt 
常见 问题 子 模块 实现 的 主要 功能 是 展示 出 数据 库 中 存储 的 有 关 编 程 中 遇 到 的 常见 问题 及 解决 方 


案 。 其 运行 结果 如 图 2.28 所 示 。 


问题 : 如何 解 岂 输出 数 奖 库 中 数据 乱码 的 且 题 ? 
在 连接 数据 库 的 文件 中 , 指定 涩 据 库 中 数据 的 蝙 码 档 式 为 "sb2312” 
这 里 连接 一 个 用 户 名 为 root, 密码 为 root 的 数据 库 服务 器 , 指定 连接 的 数据 库 为 4b_root, 并 且 指定 数据 库 
中 编码 稿 式 为 sb2312 


解 答 : 。 Seconn=nysql_connect locslhost", "root”, "root”) 
mysql_select_ab("db_root”, $conn) 
mysql_query Cset nanes eh2312") 
》 


通过 该 方 法 ,就 可 以 避免 在 输出 数 括 库 中 的 数据 时 出 现 天 码 


图 2.28 常见 问题 模块 的 运行 结果 


该 模块 由 两 个 文件 组 成 ， 一 个 是 cjwt.php 文件 ， 用 于 存储 创建 问题 数据 ， 详 细 内 容 可 以 参考 2.9.2 
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节 ; 另 一 个 是 lookcjwtphp 文件 ,用 于 输出 cjwtphp 文件 中 对 应 问题 的 详细 介绍 和 解决 方案 。 其 代码 如 下 : 
例 程 07 代码 位 置 光盘 \TM\02\bcty365\lookcjwt.php 


<?php 
include_once("conn/conn.php"): // 与 数据 库 建立 连接 
include_once("function.php"):; // 调 用 自 定义 函数 


// 读 取 tb_cjwt 表 中 的 数据 ,条 件 为 id="$_GET[id]" 

S$sql=mysql query("select * 位 om tb cjwt where id=".$ GET["id"]."".$conn): 
$info=mysql fetch array($sq)); 

> 


<table width="635" height="100" border="0" align="center" cellpaddine="0" cellspacing="0"> 
<t> 


<td width="94" height="30"><div align="center"><strong> 问 &nbsp:&nbsp: 题 : </strong></div></td> 
<td width="541"><?php echo unhtml(Sinfo["question"]): // 输 出 问题 的 详细 内 容 ?></td> 
</u> 
<t> 
<td height="70"><div align="center"><strong> 解 &nbsp:&nbsp: 答 : </strong></div></td> 
<td height="70">&nbsp:<?php echo unhtml($info["answer"]): // 输 出 问题 的 解决 方案 ?></td> 
</t> 
</table> 


2.9.4 客户 反馈 的 实现 过 程 


国 客户 反馈 使 用 的 数据 表 : tb_user、tb_leaveword 
客户 反馈 子 模块 为 客户 提供 一 个 反馈 意见 和 提出 要 求 的 平台 , 并 且 将 提交 的 信息 存储 到 数据 库 中 。 
其 运行 结果 如 图 2.29 所 示 。 


时 候 可 以 上 市 ? ~ 


sn: i 加 可 | 


图 2.29 客户 反馈 模块 的 运行 结果 


该 功能 只 对 本 网 站 中 的 会 员 开 通 ， 即 只 有 以 会 员 身份 登录 的 用 户 才 具 有 反馈 信息 的 权限 ， 其 中 在 
对 提交 表单 的 细节 处 理 上 使 用 JavaScript 脚本 来 验证 表单 中 的 值 是 否 为 室 ， 而 且 还 使 用 数字 验证 码 技 
术 。 对 表单 中 提交 的 数据 进行 处 理 是 在 saveleaveword.php 文件 中 完成 ， 该 文件 主要 用 于 对 表单 中 提交 
的 数据 进行 处 理 ， 将 数据 存储 到 数据 库 中 。 其 代码 如 下 : 

例 程 08 ”代码 位 置 光盘 \TM\02\bcty365\saveleaveword.php 


<?php session start(): // 初 始 化 一 个 session 变 量 
Sxym=$ POST[xyml: // 获 取 $_POST 提 交 的 值 
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这 $xyml!=$ SESSION["autonum'"]){ /判断 验证 码 是 否 正确 
echo "<script>alert( 效 验 码 输入 错误 !"):history.back0:</script>"; 
exit: 
y 
Stitle=$_POST["title"]: // 获 取 反 馈 信 息 的 标题 
$content=$_POST["content"]: /获取 反馈 信息 的 内 容 
$type=$_ POST["type"]: /获取 反馈 信息 的 类 型 
include_once("conn/conn.php"): /与 数据 库 建立 连接 
$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc"]."",$conn): // 读 取 数 据 库 中 数据 
Sinfo=mysql_fetch_array($sq]): /获取 结果 集中 的 数组 
$userid=$info["id"]: 
// 向 数据 库 中 添加 数据 


if(mysql query("insert into tb_leaveword(userid,type,title,.content.createtime) 
values('$userid','$type','Stitle','$content',".date("Y-m-j H:i:s").")",.Sconn)){ 
echo "<script>alert(' 留 言 发 表 成 功 ""):history.back0:</script>"; // 添 加 操作 成 功 ， 给 出 提示 信息 
jelse{ 
echo "<script>alert(' 留 言 发 表 失 败 !"):history.back0:</script>"; // 添 加 操作 失败 ， 给 出 提示 信息 
D 


?> 
2.9.5 ”单元 测试 


在 对 客户 反馈 模块 进行 测试 的 过 程 中 ， 当 提交 反馈 信息 时 ， 提 示 “ 客 户 反馈 发 表 失 败 ! ”。 经 过 
初步 的 判断 ， 问 题 可 能 出 现在 向 数据 库 添 加 数据 的 过 程 中 。 为 进一步 查找 出 问题 的 根源 ， 在 
saveleaveword.php 页 中 应 用 了 mysql_error0 函 数 ， 该 函数 可 获取 详细 的 错误 信息 ， 将 它 放置 在 insert 添 
加 语句 之 后 ， 然 后 重新 运行 网 页 ， 运 行 结果 如 图 2.30 所 示 。 


ET TE TE La 
她 址 四 ) | 怖 htp //192 168 1 59/ar/bety385/sevelesverord 了 | 园 竺 到 链接 


CTT 
图 2.30 应 用 mysql_error0 函 数 获取 的 错误 信息 


在 该 结果 图 中 可 以 看 出 是 数据 表 中 的 type 字段 出 了 问题 , 经 过 仔细 检查 发 现 , 原来 是 在 编写 insert 
添加 语句 的 过 程 中 ， 将 语句 中 字段 的 排列 顺序 与 对 应 的 字段 值 的 顺序 弄 错 了 ， 如 图 2.31 所 示 。 


OE 
if (nysql_query ("insert into tb_leaveword(userid ty 


values( $uskeid $title ,Stone 
echo “<script>alert ( 留言 发 表 成 功 !“) ;history. back() :</scfipt> : 


和 san。 ksoriptyslert( 留言 发表 类 1 ") history back () :</script 
图 2.31 代码 中 的 错误 截图 
调整 变量 $title 和 $type 的 顺序 ， 重 新 运行 程序 ， 提 示 “ 留 言 添加 成 功 ”。 
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2.10 在 线 订 购 模块 设计 


在 线 订购 模块 的 功能 是 实现 在 线 购买 企业 推出 的 软件 产品 ， 其 操作 的 流程 主要 通过 购物 车 和 订单 
管理 来 实现 。 
2.10.1 在 线 订购 模块 概述 


在 线 订 购 的 功能 对 所 有 访问 网 站 的 人 开放 ， 没 有 任何 的 权限 限制 。 其 操作 流程 如 图 2.32 所 示 。 


更 改 数量 


图 2.32 在 线 订购 模块 的 操作 流程 


2.10.2 在线 订购 模块 技术 分 析 


浏览 者 


在 线 订购 管理 模块 中 ， 不 可 缺少 的 一 项 内 容 就 是 对 订单 进行 打印 。 下 面 就 来 讲解 一 下 订单 打印 功 
能 的 实现 方法 。 在 线 订 购 管理 模块 中 运用 的 是 WebBrowser 打印 方法 。 WebBrowser 是 I 内 置 的 浏览 器 
控件 ， 无 须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 生成 ， 减 轻 服务 器 负荷 ， 缺点 是 源 文 
档 的 分 析 操 作 复杂 ， 并 且 要 对 源 文档 中 要 打印 的 内 容 进行 约束 。 

下 面 介绍 WebBrowser 控件 的 具体 参数 ， 如 表 2.4 所 示 。 


表 2.4 WebBrowser 控件 的 具体 参数 说 明 


参数 名 称 说 了 明 
document.all. WebBrowser.Execwb(7.1): 表示 打印 预览 
document.all. WebBrowser.Execwb(6.1): 表示 打印 
document.all. WebBrowser.Execwb(6.6): 表示 直接 打印 
document.all. WebBrowser.Execwb(8.1): 表示 页 面 设置 
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续 表 
参数 名 称 说 了 明 
document.all. WebBrowser.Execwb(]1.1): 打开 页 面 
document.all. WebBrowser.Execwb(2.1): 关闭 所 有 打开 的 正 窗口 
document.all.WebBrowser.Execwb(4.1): 保存 网 页 
document.all. WebBrowser.Execwb(10.1): 查看 页 面 属性 
document.all. WebBrowser.Execwb(17.1): 全 选 
document.all.WebBrowser.Execwb(22.1): 刷新 
document.all. WebBrowser.Execwb(45.1): 关闭 窗 体 无 提示 


该 技术 的 实现 原理 是 :首先 通过 onClick 事件 调用 一 个 JavaScript 脚本 ,然后 执行 openprintwindow() 
函数 ， 将 指定 的 变量 值 传递 到 订单 打印 页 中 (printwindow.php) ， 最 后 在 订单 打印 页 中 实现 打印 及 打 
印 预览 的 功能 。 调 用 JavaScript 脚本 和 执行 openprintwindow0 函 数 在 shopping_dd.php 页 中 完成 。 其 关 
键 代码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\02\bcty365\shopping_dd.php 
<script language="javascript"> 
function openprintwindow(x.y.z){ /定义 一 个 函数 ， 获 取 传 递 的 参数 
/通过 window 对 象 中 的 open 方 法 打开 一 个 新 窗口 ， 并 设置 其 属性 
© window.open("printwindow.php?ddno="+x+"&pv="+z."newframe","top=200,left=200,width=635, 
height="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=n0"); 
} 
</script> 
A -- 通 过 onclick 事 件 调用 Javascript 肢 本， 传递 参数 值 。 当 参数 z 的 值 为 "p" 时 执行 打印 功能 ----------- -- > 
<td width="75"> <img src="images/bg_14(11).jpe" width="69" height="20" 
四 onclick="javascriptxopenprintwindow('<?php echo base64_decode($ GET["ddno"]):?>".'<?php echo $egnum:?>",'p")" 
style="cursor:hand"/></td> 
<!-- ------ 通 过 onclick 事 件 调用 JavaScript 脚 本 ， 传 递 参 数值 。 当 参数 z 的 值 为 "vw" 时 执行 打印 预览 功能 --------- --> 
<td width="90"><img src="images/bg_14(12).jpe”" width="69" height="20" 
onclick="javascript:openprintwindow('<?php echo base64_decode($_GET["ddno"]):2?>'"<?php echo $egnum:?>",'v")" 
style="cursor:hand"/></td> 


< 代码 贴 十 
@ open0: 打开 一 个 窗口 ， 可 以 设置 其 URL、 名 称 、 大 小 、 按 钮 以 及 其 他 属性 。 该 方法 的 基本 语法 结构 为 : 
window.open(url,name,features.replace) 
Wl 是 要 在 新 窗口 中 打开 的 文档 的 URL 地 址 ; name 是 要 打开 的 窗口 的 名 字 ， 用 HTML 链接 的 target 属性 进行 定位 
时 会 有 用 ; features 是 一 个 用 过 号 分 隔 的 字符 囊 , 列举 窗口 的 特征 ; replace 是 一 个 可 选 的 Boolean 值 , 指出 是 否 允 许 URL 


替换 窗口 的 内 容 。 
@ onclick0: 表明 某 元 素 被 筷 标 单 击 触发 的 事件 。 


订单 的 打印 和 打印 预览 的 功能 在 printwindow.php 页 中 完成 ， 首 先 编写 一 个 实现 打印 预览 功能 的 
JavaScript 脚本 ， 然 后 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控 件 ， 最 后 获取 变量 传递 的 值 ， 当 
变量 的 值 为 p 时 执行 打印 功能 ， 当 变量 的 值 为 v 时 执行 打印 预览 的 功能 。printwindow.php 页 的 关键 代 
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/定义 一 个 函数 
/执行 WebBrowser 控 件 ， 实 现 打印 预览 
/关闭 窗口 
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码 如 下 : 
例 程 10 代码 位 置 光盘 \TM\02\bcty365\printwindow.php 
<script> 
function printviewO{ 
© document.all.WebBrowserl.ExecWB(7,1) : 
window.close():; 
上 
</script> 
ls ee 


<object ID="WebBrowserl' WID 


TH=0 HEIGHT=0 


CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 


Es = 


© <body topmargin="0" leftmargin="0" bottommargin="0" onLoad=" 


<?php 


if(s_GET["pv"}—"p"){ 


?> 


@ window.print(): 


<?php 


// 判 断 当 变 量 值 为 "p" 时 ， 执 行 打 印 操作 


建立 HTML 的 object 标 签 ， 调 用 WebBrowser 控 件 --------------------------- A 


-在 body 中 调用 onload 事 件 ， 执 行 打印 操作 > 


}elseif($_GET["pv"] 一 "v"){ /判断 当 变 量 值 为 "v" 时 ， 执 行 打印 预览 操作 


?> 


Printview0 


<?php }》 ?> 


“> 


< 人 代码 贴 二 


@ document.all.WebBrowserl.ExecWB(7,1): WebBrowser 控件 中 的 参数 ， 表 示 打 印 预览 。 
@ onLoad: 表明 某 对 象 已 载 入 窗口 。 
@ print0: window 对 象 中 的 一 个 方法 ， 实 现 打印 的 功能 。 


订单 打印 操作 的 运行 结果 如 图 2.33 所 示 。 


轨 打 印 
bi 
过 gp 


二 加 打印 机 
| 总 E7507 Stylus C85 saries 在 明日 公共 立 件 严 上 


奈 震 要 支付 的 全 向 总 计 为 


部 费 : 120 上 5 元 


杖 才 :。 礁 备 部 弹 首 克 项 四 | 
i Fs [mi ml ml ml mk 
涯 量 小 计 ( 元 ) 
厂 打印 要 文件 中 1 2000. 00 
页面 页 ET 
全 部 四 ) 全 数 亿 : [| 
(hy) 
TGF 
输入 页 到 或 责 面 范围 。 如 ,5-12 和 
取消 有 


2.33 订单 打印 操作 的 运行 结果 
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2.10.3 ”购物 车 的 实现 过 程 


国 。 购物 车 使 用 的 数据 表 : tb_bccd 
购物 车 的 功能 是 临时 储存 用 户 选 购 的 商品 ， 用 户 可 以 对 购物 车 中 的 商品 进行 添加 、 修 改 、 删 除 和 
更 新 操作 ， 也 可 以 选择 进行 结算 。 其 运行 的 结果 如 图 2.34 所 示 。 


为 禾 当 前 的 位 置 > 购物 车 


| 两 品名 称 单价 (元 ) | RT) | 人 E 
23 333.00 [De 
593 339.00 三 ”| 本 | 册 队 这 项 
《4 继续 购买 清空 购物 车 丙 品 全 敬 怠 计 : 656.00 元 EEEB 


图 2.34 购物 车 的 运行 结果 图 


购物 车 功能 实现 的 第 一 步 是 为 想 要 购买 产品 的 用 户 分 配 一 辆 购物 车 ， 使 其 能 够 记录 自己 已 经 选 购 
的 产品 。 其 工作 的 原理 与 超市 中 顾客 使 用 购物 车 进行 购物 是 相同 的 ， 只 是 这 里 使 用 的 不 是 真正 意义 上 
的 购物 车 ， 而 是 两 个 session 变量 ， 一 个 用 于 存储 用 户 选 购 商 品 的 ID 〈$goodsid) ， 另 一 个 用 于 存储 用 
户 选 购 该 商品 的 数量 〈$goodsnum) 。 如 果 用 户 在 一 次 购物 中 选 购 多 种 不 同类 的 商品 ， 则 使 用 “@” 对 
不 同类 商品 的 不 同 ID 和 数量 进行 分 隔 。 例 如 ， 用 户 选 购 的 不 同类 商品 id 为 1、2、3， 则 session 变量 
$goodsid 中 存储 的 值 为 “1@2@3@”; 其 中 同一 种 商品 不 能 购买 两 次 ， 如 果 想 要 购买 多 个 同 种 产品 ， 
可 以 在 购物 车 中 更 改 购买 商品 的 数量 。 在 本 案例 中 ， 购 物 车 的 分 配 功能 通过 shopping_cart_first.php 文 
件 来 完成 ，shopping_cart_first.php 文件 的 代码 如 下 : 

例 程 11 代码 位 置 ， 光盘 \TM\02\bcty365\shopping_cart_first.php 


@ <?php session start0; // 初 始 化 session 变 量 
四 session register("goodsid"); // 创 建 一 个 session 变 量 
@ session register("goodsnum"); // 创 建 一 个 session 变 量 
if$_SESSION["goodsid"]=="" &&: $_SESSION["goodsnum"] 一 ""){ /判断 session 变 量 中 的 值 是 否 为 空 
$_SESSION["goodsid"]=$_GET["id"]."@": /如 果 为 空 则 将 商品 的 ID 赋 给 变量 
$_SESSION["goodsnum"]="1@": // 将 商品 数量 设置 为 1@ 
}else{ 
@ Sarray=explode("@".$_SESSION["goodsid"]): /如 果 不 为 空 ， 则 使 用 @ 分 割 不 同 的 商品 ID 
/判断 如 果 获 取 的 ID 在 session 变 量 中 已 经 存在 ， 则 提示 该 商品 已 经 被 放 入 购物 车 
© if(in array($_GET["id"],$array){ 
echo "<script>alert(' 该 编程 词典 已 经 被 放 入 购物 车 ! "):history.backO:</script>"; 
exit: 
> 
$_ SESSION["goodsid"].=$_GET["id"]."@": /为 session 变 量 赋 值 
$_SESSION["goodsnum"].="1@": // 为 session 变 量 赋 值 


} 

// 将 商品 放 入 购物 车 中 ， 并 跳 转 到 购物 车 页 

echo "<script>window.location.href='shopping_cart.php':</script>": 
?> 


100 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


< 代码 贴 二 

@ session start0: 初始 化 session 变量 。 

@ session register(): 创建 一 个 session 变量 goodsid， 存 储 商 品 ID。 

四 session register0: 创建 一 个 session 变量 goodsnum， 存 储 购买 商品 数量 。 

@ explode0: 将 字符 串 依 指定 的 字符 串 或 字符 进行 分 隔 。 返 回 由 字符 事 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子囊 ， 它 
们 被 字符 囊 separator 作为 边界 点 分 隔 出 来 。 如 果 设 置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 将 包含 
string 的 剩余 部 分 ; 如 果 separator 为 空 字符 囊 (" ) ，explode0 函 数 将 返回 false; 如 果 separator 所 包含 的 值 在 string 中 找 不 到 ， 
那么 explode0 函 数 将 返回 包含 string 单个 元 素 的 数组 ;如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 

日 in array 0: 在 指定 的 数组 中 搜索 某 个 值 ， 如 果 找 到 则 返回 true， 否 则 返回 false。 


在 实现 购物 车 的 分 配 和 添加 商品 的 功能 后 ， 接 下 来 要 做 的 就 是 查看 购物 车 中 的 商品 ， 即 实现 购物 
车 中 商品 展示 的 功能 。 在 购物 车 的 商品 展示 中 ， 可 以 实现 很 多 的 操作 ， 如 清空 购物 车 、 删 除 购买 商品 、 
更 改 购买 商品 数量 、 继 续 购 物 和 结算 。 

购物 车 商品 展示 的 功能 主要 通过 shopping_cart.php 文件 来 完成 ， 首 先 从 session 变量 中 读 取 商 品 的 
ID 和 数量 , 然后 根据 商品 的 ID 循环 输出 购物 车 中 的 商品 , 最 后 以 商品 人 为 标识 符 设置 不 同 的 超 链接 ， 
执行 删除 商品 或 者 更 改 购买 商品 数量 等 操作 。 其 代码 如 下 : 

例 程 12 ”代码 位 置 : 光盘 \TM\02\bcty365\shopping_cart.php 

<table width="720" height="47" border="0" align="center”" celipadding="0" cellspacing="1" bgcolor="#999999"> 


<?php 
Sarray=explode("(@",$_SESSION["goodsid"]): // 读 取 session 变 量 中 的 商品 ID， 以 @ 进 行 分 隔 
S$arraynum=explode("@".$_ SESSION["goodsnum"]): 。“// 读 取 session 变 量 中 的 商品 数量 ， 以 @ 进 行 分 隔 
Smarkid=0; // 创 建 变 量 ， 初 始 值 为 0 
for($i=0;$i<count($array):$iH+) { // 应 用 for 循 环 语句 循环 输出 商品 ID 的 值 
if($array[$i]!=""){ /判断 如 果 商 品 ID 的 值 不 为 空 
S$markid++: /增加 变量 Smarkid 的 值 
} 
这 $markid 一 0){ // 判 断 如 果 变 量 Smarkid 的 值 为 空 则 输出 下 面 的 内 容 
?> 
<t> 


<td height="22”colspan="4”bgcolor="#FFFFFF"><div align="center"> 对 不 起 您 的 购物 车 中 暂 无 商品 信 
息 !</div></td> 


</t> 
<?php 
jelse{ // 如 果 $markid 的 值 不 为 空 ， 则 执行 下 面 的 内 容 
Stotalprice=0: /创建 变量 Stotalprice， 初 始 值 为 0 
for($i=0:$i<count($array): SiH+){ /循环 输出 数组 中 的 商品 ID 值 
if($array[Si]!=""){ 
/根据 获取 的 商品 ZDP 的 值 ， 从 数据 库 中 获取 对 应 产品 的 信息 
$sqlcart=mysql_ query("select * from tb_bccd where id=".$array[$i]."".$conn): 
$infocart=mysql fetch array($sqlcart) 
pe 
<t> 


<form name="form<?php echo $array[$i]?>" method="post" action="changegoodsnum.php"> 
<td height="22" bgcolor="#FFFFFF">&nbsp:<?php echo unhtml(S$infocart["bccdname"]):?></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><?php echo 
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number format($infocart["price"].2):?></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><input type="text" name="goodsnum" 
value="<?php echo $arraynum["$i"]:?>" class="inputcss" size="8" ><input type="hidden" name="id" value="<?php echo 
Sinfocart["id"]:?>" ></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><a href—"javascript:form<?php echo 
$array[$i]?>.submit():" class="al"> 更 改 数量 </a>&nbsp:|&nbsp:<a href="delgoods.php?id=<?php echo $infocart["id"]:?>" 
class="al"> 删 除 该 项 </a></div></td> 


</form> 
</t> 
<2php 
Stotalprice+=S$infocart["price"]*$arraynum["$i"]; 
} 
b 
} 
?> 
</table> 


2.10.4 ”商品 订单 的 实现 过 程 


国 ”商品 订单 使 用 的 数据 表 : tb_bccd、tb_dd、tb_city 

在 确定 所 要 购买 的 商品 之 后 ， 接 下 来 要 做 的 就 是 进行 购物 结算 ， 填 写 用 户 购物 订单 ， 将 订单 保存 

到 数据 库 中 ， 并 且 随 机 生成 一 个 订单 号 ， 作 为 订单 的 唯一 标识 。 生 成 订单 的 运行 结果 如 图 2.35 所 示 。 
寺村 省 明 日 科技 有 限 公司 -编程 司 奥 订 单 2007-12-02 16:47:59 

订 竺 号 : 07120216774091 


部 篇 ; 130000 
固定 电话 ; 0431-64978981 


联系 地 址 。 长春 市 


商品 名称 单价 (元 》 数量 人 (元 ) 
333 23.00 1 353.00 
ED 3 0 i 353.00 
商品 总 计 : 665.00 元 
邮费 ; 240 10 元 


你 党 要 支付 的 爹 征 总 计 为 : 906 10 元 


图 2.35 生成 订单 的 运行 结果 图 
订单 提交 以 后 用 户 可 以 选择 汇款 的 方式 : 一 是 选择 网 上 支付 ， 那 么 将 跳 转 到 企业 指定 的 网 上 银行 
进行 汇款 ， 汇 款 的 操作 将 在 企业 指定 的 网 上 银行 中 进行 ， 这 里 不 做 讲解 ; 二 是 选择 到 指定 的 银行 向 企 
业 提供 的 账号 中 汇款 。 企 业 将 在 收 到 汇款 后 按照 用 户 指定 的 地 址 和 方式 将 产品 送 到 。 商 品 订单 的 生成 
和 处 理由 shopping_cart_getuserinfo.php 和 savebuyuser.php 文件 来 完成 。 
订单 处 理由 savebuyuser.php 文件 完成 ， 首 先 连 接 数据 库 ， 随 机 生成 一 个 订单 号 ， 然 后 获取 购物 车 
中 的 商品 信息 ， 最 后 将 商品 信息 和 订单 号 存储 到 数据 库 中 。 其 代码 如 下 : 
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例 程 13 ”代码 位 置 : 光盘 \TM\02\bcty365\savebuyuser.php 


<?php session start(): // 初 始 化 session 变 量 
include_once("conn/conn.php"): // 连 接 数 据 库 
$ddnumber=substr(date("YmdHis").2.8).mt_rand(100000.999999); // 随 机 生成 订单 号 


$sql=mysql_query("select * from tb_city where id=".$ POST["city"]."".Sconn): // 读 取 数 据 库 中 的 城市 信息 
$info=mysql fetch array($sq)); 
if($shfs—"1"){ // 判 断 用 户 选择 的 送 货 方式 
Syprice=S$info[pt]: 
Sshfs=" 普 通 邮 递 "; 
}elseif($shfs=—"2"){ 
Syprice=Sinfo[kd]: 
Sshfs=" 邮 政 特快 专递 EMS"; 
} 


$array=explode("(@",$_SESSION["goodsid"]):; 


// 以 @ 来 分 隔 session 变 量 中 存储 的 商品 人 D 


Sarraynum=explode("@",$_SESSION["goodsnum"]): // 以 @ 来 分 隔 session 变 量 中 存储 的 商品 数量 


S$totalprice=0; 
for($i=0;$i<count($array):$i++){ 
if($array[$i]!=""){ 


// 循 环 读 取 数组 中 商品 的 人 D 


$sqlcart=mysql query("select * 位 om tb_bccd where id=".$array[$i]."",$conn): 
$infocart=mysql_fetch_array($sqlcart): 
Stotalprice+=Sinfocart["price"]*S$arraynum["S$i"]; 
! 
} 
Stotalprice=$totalpricet+ $yprice: // 获 取 汇 款 金额 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


if(mysql_query("insert into 
tb_dd(ddnumber.,recuser,sex.address.yb.qq.email.mtel.gtel.shfs.spc.slc,yprice.totalprice,createtime.cityid) 
values(".$ddnumber.",".$_ POST["recuser"].",".$_POST["sex"].",".$_POST["address"].",".$_POST["yb"].",".$_POST["qq" 
].",".$_POST["email"].",".$_POST["mtel"].",".$_POST["gtel"].".".$shfs.",".$_SESSION["goodsid"].".".$_SESSION["goo 
dsnum"].",".$yprice.",".$totalprice.",".date("Y-m-d H:i:s").".".$_POST["city"].")",Sconn) { 
session_unregister("goodsid"): // 注 销 session 变 量 goodsid 
session_unregister("goodsnum"): // 注 销 session 变 量 goodsnum 
echo "<script>window.location.href='"shopping_dd.php?ddno=".base64_encode($ddnumber).";</script>"; 
}else{ 
echo "<script>alert(' 订 单 信息 保存 失败 ， 请 重 试 !"):</script>":; 


} 


?> 


2.10.5 ”单元 测试 


在 在 线 订购 模块 的 测试 过 程 中 ， 当 执行 删除 购物 车 中 某 个 商品 的 操作 时 ， 发 现 该 项 操作 没 能 正确 
执行 ， 指 定 的 商品 没有 被 删除 。 分 析出 错 原因 : 首先 ， 从 该 功能 实现 的 思路 入 手 ， 并 且 和 仔细 核对 其 中 
使 用 的 变量 以 及 函数 是 否 正 确 。 

删除 商品 的 操作 是 根据 商品 的 ID 来 进行 处 理 的 ， 当 单 击 “ 删 除 ” 超 链 接 时 ， 将 指定 商品 的 ID 通 
过 变量 $ides 传 入 到 执行 删除 商品 操作 的 文件 delgoods php 中 ， 然 后 在 delgoods .php 文件 中 ， 获 取 变 量 
$ides 传 入 的 ID 值 ， 并 且 根 据 ID 值 完成 删除 购物 车 中 指定 商品 的 操作 。 在 查看 该 功能 实现 的 过 程 中 发 
现 ， 定 义 的 变量 $ides 与 delgoods.php 文件 中 $_GET["id"] 获 取 的 变量 值 不 一 致 ， 从 而 导致 删除 商品 的 操 
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作 没有 正确 执行 ， 代 码 的 出 错位 置 如 图 2.36 所 示 。 
shopping_cart php 文 件 中 执行 天 除 探 作 的 内 容 


de 22” bgcolor="#FFFFFF"> 
aligr=" center” 


echo Cad esp i ”class="al") 更 改 数量 4/a> 
d"]; 


7》” class=al" 删除 该 项 </ay 


delgoods. phi 文件 执行 除 控 作 的 内 容 
HT 

品 的 人 D 
“go0dsid"]) RE 

委 组 中 搜索 指证 的 值 , 各 县 万 则 返回 相应 的 刍 名 
/的 庙 妆 旺 区 久生 人 
ON[" goodsid”]=inplode ("@", $arrayid) // 交 充 量 的 值 添加 到 zezsior 谢 量 中 


$- goodsnum ]=inplode("@ Sarraynum) 
san ”kscriptywindow, location. href=" shopping_cart. php’ ;</script)” 


图 2.36 代码 中 的 错误 显示 
查找 出 错误 原因 ， 将 定义 的 变量 与 获取 的 变量 值 统 一 使 用 id， 保 存 文件 ， 重 新 运行 程序 ， 删 除 操 
作 可 以 正常 运行 


2.11 社区 论坛 模块 设计 
社区 论坛 模块 为 网 站 的 浏览 者 提供 一 个 交流 的 平台 ， 以 此 来 扩大 网 站 的 影响 力 ， 汇聚 更 多 的 人 气 ， 
宣传 企业 形象 ， 推 广 企业 产品 。 


2.11.1 社区 论坛 模块 概述 


社区 论坛 模块 为 浏览 者 、 会 员 、 客 户 和 企业 之 间 提 供 一 个 大 的 交流 平台 ， 根 据 身 份 的 不 同 ， 给 予 
分 别 拥有 不 同 的 操作 权限 ， 社 区 论坛 模块 的 操作 流程 如 图 2.37 所 示 。 


ES 


图 2.37 社区 论坛 流程 图 
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在 本 论坛 中 ， 浏 览 者 只 能 够 查看 帖子 ， 注 册 会 员 既 可 以 查看 帖子 ， 也 可 以 发 布 和 回复 帖子 ， 管 理 
员 则 具有 发 布 、 回 复 、 查 看 和 删除 的 权限 。 


2.11.2 社区 论坛 模块 技术 分 析 


在 社区 论坛 模块 的 实现 过 程 中 , 通过 JavaScript 脚本 和 下 拉 列 表 框 的 结合 实现 一 个 不 同 版 块 之 间 快 
速 跳 转 的 功能 ， 从 而 能 够 更 加 灵活 、 方 便 地 实现 不 同 版 块 之 间 的 跳 转 。 

下 面 分 析 该 技术 是 如 何 实现 的 ， 该 技术 的 实现 综合 3 个 方面 的 内 容 ， 以 一 个 下 拉 列 表 框 为 主 ， 通 
过 PHP 语句 从 数据 库 中 读 取 数据 作为 下 拉 列 表 框 的 值 ， 应 用 onchange 事件 来 调用 JavaScript 脚本 ， 实 
现 不 同 版 块 之 间 的 跳 转 。 这 里 以 bbs_top.php 文件 中 的 快速 跳 转 功能 为 例 进行 分 析 。 其 关键 代码 如 下 : 

例 程 14 ”代码 位 置 ， 光盘 \TM\02\bcty365\bbs_top.php 

<!-- 创 建 一 个 下 拉 列 表 框 ， 指 定名 称 为 select type， 并 且 设 置 其 属性 ， 通 过 onchange 事 件 来 调用 JavaScript 脚 本 文件 ， 
实现 页 面 跳 转 --> 

<select name="select type" class="inputcss" 
© onChange="javascript:window.location=this.options[this.selectedIndex].value;" > 


<?php 
// 通 过 PHP 语 句 从 数据 库 中 读 取 数据 ， 使 用 数据 的 ID 作为 下 拉 列 表 框 的 值 ， 使 用 数据 的 标题 title 作 为 下 拉 列 表 框 显示 的 内 容 
$sql=mysql query("select * from tb type small order by createtime desc",$conn); 
Sinfo=mysql fetch_array($sqD); 


if($info—""){ 
echo "<option> 暂 无 讨论 区 </option>"; 
}else{ 
echo "<option>- 版 块 快速 跳 转 -</option>": 
dof /应 用 do..while 循 环 语句 输出 下 拉 列 表 框 中 的 值 
© echo "<option value="bbs_list.php?id=".Sinfo[id]."'>".$info[title]."</option>"; 
} 
while($info=mysql fetch_array($sql)): // 应 用 do...while 循 环 语句 结束 
由 
?> 
</select> 
< 代码 贴 二 


@ onChange: 某 元 素 失去 焦点 ， 并 且 从 用 户 最 后 一 次 访问 以 来 ， 其 值 已 经 改变 。 
location: 用 于 访问 窗口 的 当前 定位 (URL) ， 既 可 被 读 取 ， 又 可 被 置换 ， 可 以 通过 其 实现 某 个 页 面 的 定位 或 者 更 新 。 
@ <option value= … >…</option>: 下 拉 列 表 框 中 输出 的 值 ， 以 及 显示 的 内 容 。 


该 技术 实现 的 运行 结果 如 图 2.38 所 示 ， 它 将 实现 从 硬件 查询 模块 跳 转 到 PHP 模块 。 
2.11.3 ”论坛 分 类 的 实现 过 程 
国 ”论坛 分 类 使 用 的 数据 表 : tb_bbs、tb_type_big、tb_type_small 


论坛 分 类 可 以 分 为 两 类 ， 一 个 是 论坛 中 大 的 版 块 分 区 ， 分 为 6 个 版 块 : 综合 信息 讨论 区 、 操 作 系 
统 、 程 序 设计 交流 区 、 网 管 技术 应 用 、Web 程序 开发 和 数据 库 技术 ， 其 数据 存储 于 tb_type_big 数据 表 
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中 。 另 一 个 是 对 应 不 同 的 版 块 中 不 同 语言 和 技术 的 分 类 , 分 为 11 种 , 其 数据 存储 于 tb type_small 表 中 。 
论坛 分 类 的 运行 结果 如 图 2.39 所 示 。 


(加 当前 位 置 >》 社 区 论坛 论坛 版 块 、 查 斤 帖 子 


计算 机 硬件 咨询 版 块 


国 | 阅 日 要 的 计算 机 不 能 局 动 了 ? 


共有 帖子 ! 条 每 页 显示 30 条 第 1 页 / 共 1 页 


计算 机 硬件 咨询 


创建 时 间 : 2009-12-03 00:00:00 


[FEZ3 ET 了 7 II 
py esp net 
创 哇 时 间 : 2000-12-16 00:00:00 


Php 


fA : 2009-12-05 00.00:00 


5 是 世界 上 使 用 最 广泛 计算 机 语言 。 的 面向 对 铺 的 扩展 称 为 ct+ 


fl/ ; 2005-12-05 00:00:00 


软件 工程 /管理 | Software engineering/managenent 


| ej : nin 00;00:00 
图 2.39 论坛 分 类 的 运行 结果 
论坛 分 类 的 实现 原理 很 简单 ， 首 先 从 tb_type_big 表 中 读 取 6 个 版 块 中 的 数据 ， 进 行 循环 输出 ， 然 


后 在 版 块 中 典 套 循环 ， 用 于 输出 不 用 语言 的 分 类 数据 。 该 功能 主要 通过 bbs_index.php 文件 来 完成 ， 
bbs_index.php 文件 的 程序 代码 如 下 : 


例 程 15 ”代码 位 置 光盘 \TMW2\bcty36S\bbs_index.php 


<php include once("top.php"): // 调 用 网 站 头 文件 
include_once("bbs_top.php"): // 调 用 社区 论坛 的 头 文件 

?> 

<?php 


// 循 环 输出 数据 表 tb_type_big 中 的 6 个 版 块 数据 

$sql=mysql_query("select * 位 om tb_type_big order by createtime desc".$conn): 

S$info=mysql fetch array($sq)): 

if($info—false){ // 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 


?> 
… 省 略 了 部 分 代码 


<?php 
Yelse{ // 如 果 返 回 值 为 rue， 则 执行 do…while 循 环 语句 
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卢 外 部 嵌 套 循环 ， 输 出 论坛 中 的 版 块 分 类 数据 */ 
© dof 
pe 
… 省 略 了 部 分 代码 
<table width="750" border="0" align="center" cellpadding="0" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#6EBEC7"> 
<?pl 
// 循 环 输出 tb_type_small 表 中 的 不 同 语言 和 技术 的 分 类 数据 
S$sqll=mysql query("select * from tb_type_small where bigtypeid=".$info["id"]."",$conn); 
Sinfol=mysql fetch array($sql1); 


if($infol=—false){ // 判 断 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 
?> 
… 省 略 了 部 分 代码 
<?php 
yelse{ // 如 果 返 回 值 为 rue， 则 执行 下 面 的 内 容 ， 输 出 该 版 块 中 对 应 语言 和 技术 的 帖子 详细 信息 
?> 
… 省 略 了 部 分 代码 
[2 <?php ”人 * 内 部 嵌 套 循环 ， 输 出 不 同 语言 和 技术 的 分 类 */ 
dof 2> 
<tr> 
<td height="30"><font color="#666666"> 创 建 时 间 : <?php echo $infol["createtime"]:?></font></td> 
</u> 
“省 略 了 部 分 代码 
<?php 
}while(Sinfol=mysql fetch array($sql1)); 
族 内 部 嵌 套 循环 结束 */ 
A 
</table> 
<?php 
}while(Sinfo=mysql fetch array(SsqD); 
族 外 部 嵌 套 循环 结束 ， 对 版 块 中 的 大 类 进行 输出 */ 
} 
> 
<?php include once("bottom.php"); > 
< 代码 贴 二 


@ do…while 循环 语句 ， 对 论坛 中 大 的 版 块 分 类 进行 循环 输出 。 
@ do…while 循环 语句 ， 对 论坛 中 一 个 版 块 中 的 不 同 语言 和 技术 进行 循环 输出 。 


全 注意 ; 在 应 用 do…while 循环 语句 时 ，while 后 的 分 号 不 能 省 略 。 
2.11.4 ”论坛 帖子 浏览 的 实现 过 程 
国 ”论坛 帖子 浏览 使 用 的 数据 表 : tb bbs、tb_user、tb_reply 


论坛 帖子 浏览 主要 输出 指定 帖子 的 详细 信息 ， 包 括 发 帖 人 人、 用户 级 别 和 注册 的 时 间 ， 以 及 帖子 的 
主题 、 内 容 和 发 帖 时 间 ， 包 括 上 传 的 图 片 。 本 模块 中 是 用 户 权 限 使 用 体现 的 最 明显 的 地 方 ， 可 以 分 为 
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三 种 情况 : 第 一 以 浏览 者 进行 登录 ， 只 能 是 浏览 帖子 的 内 容 ， 没 有 其 他 权限 ; 第 二 以 会 员 进 行 登录 ， 
可 以 对 帖子 进行 回复 ， 发 表 自 己 的 看 法 ; 第 三 以 管理 的 身份 进行 登录 ， 不 但 可 以 回复 帖子 ， 而 且 可 以 
对 任何 人 发 布 和 回复 的 帖子 进行 删除 和 顶 帖 的 操作 。 下 面 就 来 看 一 下 以 管理 员 身 份 进行 登录 时 都 具备 
哪些 权限 ， 运 行 结果 如 图 2.40 所 示 。 


本 加 何 配置 PCF 环境 


[@) 2007-12-12 13-19;47 
从 起 请 玫 一 下 如 何 配置 PIC 的 环境 


发 贴 人 : lx 
信箱 坊 0ic4 时 林地 址 
用 户 倒 别 ; 管理 员 


注册 时 间 : 2007-12-01 13:15:25 


发 贴 人 :lz 由 爱 主 题 : FHF 的 环境 包 置 方法 
“信箱 请 0ica 易 人 ?地 址 
用 尸 凶 别 : 管理 两 
发 贴 吕 数 ; 5 | 证 浊 内 容 请 参考 PITE 数 据 尿 系统 开发 元 全 手 骨 中 的 第 1 章 


计划 时 全 :200-i20L 13:15:05 
图 2.40 管理 员 浏览 帖子 的 结果 图 


论坛 帖子 浏览 的 功能 通过 bbs_looks.php 文件 完成 ， 首 先 根据 传递 的 ID 值 读 取 指 定 的 帖子 数据 ， 
然后 判断 登录 用 户 的 类 型 ， 最 后 根据 用 户 不 同 的 类 型 执行 不 同 的 操作 。 其 代码 如 下 : 


例 程 16 ”代码 位 置 : 光盘 \TM\02\bcty365\bbs_lookbbs.php 
<?php 
/根据 $_GET 传 递 的 数据 获取 tb_bbs 中 的 数据 
$sqlb=mysql query("select * from tb_bbs where id=".$_GET["id"]."".Sconmn): 
Sinfob=mysq] fetch_array($sqlb): 
/根据 $_GET 传 递 的 数据 获取 tb_user 中 的 数据 
$sql4=mysql query("select * from tb_user where id=".$infob["userid"]."".$conmn): 
Sinfo4=mysql fetch_array($sql4): 
?> 
// 省 略 了 部 分 HTML 代 码 
<table width="180" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="22"> 用 户 级 别 : 


<?php 
/根据 用 户 信息 表 tb_user 中 字段 usertype 的 值 判断 该 用 户 的 类 型 
// 如 果 值 为 1 则 是 管理 员 ， 值 为 2 则 是 后 台 管理 员 ， 值 为 0 则 是 普通 会 员 
© 这 Sinfo4["usertype"] 一 "1") echo "管理 员 ":else echo "普通 会 员 ": 


</table> 
本 /省 略 了 部 分 HTML 代码 
<table width="500" height="200" border="0" align="center" cellpadding="0" cellspacing="0"> 
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<tr> 
<td width="35" height="20"><div align="center"><img src="images/lt_ 15(11).jpe" width="25" 
height="25"></div></td> 
<td><?php echo $infob["createtime"]:?></td> 


</t> 
<t> 
<td height="150" colspan="2"> 
<?php 
// 判 断 tb_bbs 表 中 的 字段 photo 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
@ if(Sinfob[photo]!=""){ 
Sphotos=substr(Sinfob[photo],2,70); // 获 取 图 片 在 服务 器 中 的 存储 路 径 
echo (stripslashes(Sinfob["content"])); // 输 出 帖子 的 内 容 
echo "<img src=\"Sphotos\">"; // 根 据 获 取 的 图 片 路 径 ， 输 出 服务 器 中 的 图 片 
jelsef // 如 果 tb_bbs 表 中 的 图 片 字段 photo 为 空 ， 则 执行 下 面 的 内 容 
目 echo (stripslashes(Sinfob["content"])); /只 输出 帖子 的 内 容 
和 : 
<td> 
</t> 
</table> 
<table width="530" height="20" border="0" align="center" cellpadding="0" cellspacing="0"> 
<t> 


<td width="239" height="30">&nbsp:</td> 
<td width="291"><img src="images/lt_15(5).jpe”" width="72" height="23" style="cursor:hand" onclick=" 
<?php 
// 如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 顶 帖子 的 操作 
if($_SESSION["unc"]—""){ 
echo "javascript:alert(' 请 先 登 录 本 站 ， 然 后 进行 此 操作 ! ");window.location.href='index.php':"; 


}else{ 
// 否 则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 顶 帖 
$sqlu=mysql query("select usertype from tb_user where 
Usemc=".$_ SESSION["unc"]."".Sconmn): 
S$infou=mysql fetch array($sqlu): 


9 if(Sinfou["usertype"]—D{ // 如 果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href="settop.php?id=".Sinfob["id"™"]."""; 
jelsef // 否 则 不 具备 该 权限 
echo "javascript:alert( 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
> 
?> 
">&nbsp:&nbsp: 
[32 <?php 


// 判 断 当 前 用 户 是 否 具有 删除 帖子 的 权限 
if($ SESSION["unc"]!=""){ 
// 条 件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
Ssqlu=mysql_ query("select usertype from tb_user where 
usernc="".$ SESSION["unc"].""",Sconn); 
Sinfou=mysql fetch array(Ssqlu); 
if(Sinfou["usertype"]—D{ 
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?> 
<img src="images/lt_ 15(10).jpg" onclick="javascript-if(window.confirm(' 您 确定 删除 该 帖 吗 ? 
"=—true) {window.location.href='bbs_delete.php?id=<?php echo $infob["id"]?>":}" style="cursor:hand"/> 


<2php 
} 
} 
?> 
</td> 
</t> 
</table> 

的 代码 贴 十 


@ $info4["usertype"]: 判断 用 户 的 类 型 ， 如 果 值 为 1 是 管理 员 ， 否 则 为 普通 会 员 。 
@ $infob[photo]: 判断 发 布 的 帖子 中 是 否 含有 图 片 ， 如 果 有 则 输出 ， 没 有 则 不 输出 。 
上 @ stripslashes0: 将 应 用 addcslashesO 函 数 处 理 后 的 字符 串 返 回 原样 。 

@ 判断 登录 用 户 是 否 具有 顶 帖 的 权限 。 

上 @ 判断 登录 用 户 是 否 具有 删除 帖子 的 权限 。 


[四 说 明 : 上 面 给 出 的 是 该 文件 的 部 分 代码 ， 主 要 讲解 了 该 功能 的 实现 方法 ， 完 整 的 代码 可 以 参考 本 
书 光盘 中 光盘 \TMNWO2\bcty365\bbs lookbbs.php 文件 。 
2.11.5 “论坛 帖子 发 布 的 实现 过 程 


国 ”论坛 帖子 发 布 使 用 的 数据 表 : tb_bbs、tb_user 
论坛 帖子 发 布 通过 两 个 文件 来 完成 ， 一 个 是 帖子 发 布 信息 的 提交 页 bbs_pubs.php， 另 一 个 是 对 提 
交 的 数据 进行 处 理 的 retrieve.php 文件 。 该 功能 实现 的 运行 结果 如 图 2.41 所 示 。 


| 


帖子 主题 ; 


入 六 人 修 现 在 的 作 情 知 何 ? 
二。 


当前 用 户 : 1z 
二 信箱 七 0ica 虽 IT 地 址 
用 户 姬 别 | : 管理 员 
发 帆 总 汰 :5 
注册 时 间 : 2007-12-01 13-15:25 


Oo ne 
I 
cArceorce@r° 


为. gif] jpD) 


El wl 


2.41 帖子 发 布 模块 的 运行 结果 


在 发 布 信息 的 提交 页 中 ， 显 示 当 前 用 户 的 个 人 信息 ， 设 置 添加 数据 表单 元 素 ， 其 中 表单 元 素 的 设 
计 如 表 2.5 所 示 。 
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名 称 


表 2.5 发 布 信息 页 中 使 用 的 表单 元 素 
重要 属性 


含 义 


form bbs 


method="post" action="retrieve.php" enc "multipart/form-data" 


发 帖 表 单 


bbs_type 


select 


text 


class="inputcss" style="backeround-color:#6EBEC7"> 
<?php 

$sql=mysql query("select * from tb_type_small order by createtime 
desc",$conn): 

Sinfo=mysql fetch array($sq)); 

if($info==false){ 

echo "<option> 暂 无 讨论 区 </option>"; 

}else{ 

dof 


?> 
<option value="<?php echo $info[id] :?>"<?php 这 $_GETI[id] 一 $infofid]) 
{echo "selected=\"selected\"":}?>><?php echo S$info[title]:?></option> 
<?php } 
while($info=mysql fetch_array($sql)): 
}?> 


bbs title 


class="inputcss" style= "background-color#6EBEC7"> 


帖子 的 语 
言 或 者 技 
术 的 类 别 


帖子 标题 


bbs_head 


radio 


Value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 


表情 图 


file 


id="bbs_photo" class="inputcss" style="background-color:#6EBEC7" /> 


上 传 图 片 


contentl 


textarea 


jd="contentl" class="inputcss" style="backeround-color:#6EBEC7"> 


帖子 内 容 


Submit 


submit 


Value=" 提 交 " 


提交 表单 


在 retrieve.php 页 中 对 表单 提交 的 数据 进行 处 理 ， 将 数据 存储 到 tb_bbs 表 中 ， 并且 更 新 用 户 信息 表 
tb_user 中 pubtimes 字段 的 值 ， 其 中 还 应 用 了 图 片上 传 技术 ， 将 图 片上 传 到 服务 器 中 指定 的 文件 夹 下 。 


retrieve.php 文件 的 代码 如 下 : 

例 程 17 ”代码 位 置 : 光盘 \TM\02\bcty365\retrieve.php 
<?php session start(): /初始 化 session 变 量 
Stitle=$_POST[bbs_title]: // 获 取 帖子 的 标题 
$content=$_ POST[contentl]: // 获 取 帖子 的 内 容 
/* 判断 提交 的 帖子 主题 和 帖子 内 容 是 否 为 空 */ 
if(Stitle—""){ 

echo "<script>alert(' 请 输入 帖子 主题 !):history.backO:</script>": 

exit } 


if($content—""){ 
echo "<script>alert( 请 输入 帖子 内 容 !'"):history.backO:</script>"; 
exit; } 

dy 

include_once("conn/conn.php"): // 连 接 数 据 库 

// 根 据 $_SESSION["unc"] 的 值 读 取 数据 库 中 用 户 的 信息 

$sql=mysql_query("select * 位 om tb_user Where usemc=".$_SESSION["unc"]."".$conn); 
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Sinfo=mysql fetch array($sq]l): /检索 指定 条 件 的 数据 信息 
Suserid=$info[id]: // 获 取 用 户 id 
$typeid=$_ POST[bbs_ type]: /接收 版 块 名 称 
S$title=$_ POST[bbs title]: /接收 帖子 主题 
$content=$_ POST[content1]: /接收 帖子 内 容 
Shead=$_POST[bbs_head]: // 接 收 头 像 
Screatetime=date("Y-m-j H:i:s"): // 获 取 系 统 当前 时 间 
Slastreplytime=$createtime: // 将 当前 时 间 赋 给 变量 
Sreadtimes=0; 
Slink=date("YmjHis"): 
@ if(s FILES['bbs photo']["name"]==true){ /上 传 图 片 ， 判 断 文件 是 否 存在 ， 如 果 存 在 则 执行 下 面 的 内 容 
© S$photo_name=strtolower(stristr(S FILES["bbs photo"]["name"],".")): // 获 取 图 片 后 缀 名 ， 将 字符 转 成 小 写 
if($photo_name!=".gif" & $photo_name!=".jpg" & $photo_name!=".jpeg" ){ // 判 断 图 片 的 格式 是 否 符 合 要 求 
echo "<script>alert(' 您 上 传 的 图 片 格式 不 正确 !"):history.back0:</script>"; 
}else{ 
Spaths1=$link.mt_rand(1000000.9999999).$photo_name: // 创 建 图 片 的 名 称 
S$photos="./upfile/".$paths1:; // 创 建 图 片 的 存储 路 径 
@ Imove_uploaded file(S FILES["bbs_photo']["tmp_name"],Sphotos); /将 图 片 存储 到 指定 的 文件 夹 下 
/向 数据 库 添 加 数据 


这 mysql_query("insert into tb_bbs(userid,typeid,title,content,createtime,lastreplytime,head,readtimes,top,photo) 
values("".$userid.",".$typeid.",". $title.",".$content.",".$createtime.",". $lastreplytime.",".$head.",".$readtimes.",'0','$photos' 
)".Sconn){ 

mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); // 更 新 tb_user 中 pubtimes 字 段 的 值 
echo "<script>alert(' 新 帖 发 表 成 功 !");history.backO:</script>"; 

j}else{ 

echo "<script>alert( 新 帖 发 表 失 败 !"):history.backO:</script>"; 
} 


} 
yelse{ // 如 果 没有 提交 图 片 ， 则 执行 下 面 的 内 容 
这 mysql_query("insert into tb_bbs(userid,typeid,title,content,createtime.lastreplytime.head,readtimes,top) values 
(".$userid.",".$typeid.".". $title.".".S$content.".".$createtime.".".$lastreplytime."."".$head.",".$readtimes.",'0")".$conn)){ 

mysql_query("update tb_user set pubtimes=pubtimes+1".$conn): 
echo "<script>alert( 新 帖 发 表 成 功 !"):history.back0 〇 :</script>": 

}else{ 
echo "<script>alert( 新 帖 发 表 失败 !"):history.backO:</script>": 

} 


} 
?> 


< 代码 贴 十 

@ $ FILES[bbs photo]["name"]: $ FILES[] 全 局 变量 ， 获 取 表 单 提交 文件 的 原始 名 称 。 

@ strtolower0: 将 指定 的 字符 转换 为 小 写字 母 。 

stristr0: 获取 指定 字符 囊 ( A) 在 另 一 个 字符 串 (B ) 中 首次 出 现 的 位 置 到 (B ) 字符 囊 末 尾 的 所 有 字符 囊 。 该 函数 
如 果 执 行 成 功 则 返回 剩余 的 字符 囊 ， 否 则 将 返回 false。 

@@ mt rand0: 生成 一 个 随机 数 ， 用 于 上 传 文件 的 名 称 。 

@ move uploaded file0: 将 指定 的 文件 上 传 到 指定 的 文件 夹 下 。 
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2.11.6 


论坛 帖子 回复 的 实现 过 程 


国 ”论坛 帖子 回复 使 用 的 数据 表 : tb bbs、tb user、tb reply 


回复 论坛 中 的 帖子 ， 必 须 是 以 会 员 或 者 管理 员 的 身份 进行 登录 ， 否 则 不 能 进行 帖子 的 回复 操作 ， 


其 运行 结果 如 图 2.42 所 示 。 


也 TF 编程 同 典 , 程 计 开发 刘斌 


© 200r-l1-39 tl2:44 
© un 


发 同人 :TSoft 
信箱 仿 0;c4 时 本 地 址 
己 凶 别 ; 管 通 会 
pe 欢 四 大 守 提 出 主 吐 意见 ! 
发 贴 总 数 :4 


让 央 时 间 : 2007-11-18 08.42:43 a CD 
[a 回复 主题 : 


入 
, 图 片 : Veracleebp eee nr tbety Dl 四 片 不 Rbgidzm 格式 为 tif/ jpg 
人 : TSofv 
信箱 苞 0ic4 号 IT 地 址 本 
用 尸 研 别 ; 葡 通 会 员 
已 数 J 


| En :mtime 提交 | 重 填 | 
图 2.42 论坛 帖子 回复 的 运行 结果 


论坛 帖子 回复 功能 的 实现 主要 通过 bbs looks.php 和 savereply.php 两 个 文件 完成 。 其 中 应 用 
JavaScript 脚本 对 回复 帖子 的 文本 框 进行 输出 和 隐藏 的 控制 。 在 bbs_looks.php 文件 中 , 帖子 回复 使 用 的 
表单 元 素 如 表 2.6 所 示 。 


form reply 


表 2.6 论坛 帖子 回复 中 的 重要 表单 元 素 


method="post" action="savereply.php" enctype="multipart/form-data"> 


reply_title 


回复 帖子 主题 


class="inputcss" id="reply_title" 


bbsid hidden value="<?php echo Sinfob["id"]:?>" 对 应 帖子 的 JD 
bbs_ head radio | Value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
bbs_photo file | id="bbs_photo" class="inputcss" 上 传 图 片 
contentl textarea id="content1" 回复 帖子 内 容 


Submit 


value=" 提 交 " 提交 表单 


submit 


在 帖子 回复 表单 bbs_looks.php 页 中 ， 首 先 判断 登录 用 户 是 否 具有 回复 的 权限 ， 然 后 根据 提交 的 值 
展开 回复 表单 的 文本 框 ， 在 文本 框 中 输入 回复 的 主题 和 内 容 ， 最 后 将 数据 提交 到 表单 处 理 页 savereply.php 
中 。bbs_looks.php 的 主要 代码 如 下 : 
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例 程 18 代码 位 置 : 光盘 \TM\02\bcty365\ bbs_lookbbs.php 
<script language="javascript"> 


// 设 计 回 复 帖子 表格 的 输出 方式 
function show_replyO{ // 定 义 一 个 函数 
ifreply_ bbsl.style.display=="") // 判 断 当 display 的 值 为 空 时 
reply_bbsl.style.display="none" ; // 则 输出 表格 
button_show_bbs.value=" 回 复 帖 子 "; // 显 示 回 复 帖子 
} 
else if(reply_bbsl.style.display—"none") // 判 断 当 display 的 值 为 none 时 
{ 
reply_bbsl.style.display="" : // 则 不 输出 表格 
button_show_bbs.value=" 关 闭 窗 口 "; // 显 示 关闭 窗口 
3 
} 
</script> 
< 判断 登录 用 户 是 否 具有 回复 的 权限 ------------------------------------------ -- > 
<img src="images/lt_15(9).jpg" width="72" height="23" id="button_show_ bbs" style="cursor:hand" onClick=" 
<?php 
if($_SESSION["unc"]=—""){ 
echo "javascript:alert(' 请 先 登 录 本 站 ， 然 后 回复 帖子 !):window.location.href='index.php';"; 
}else{ 
es 
show_replyO 
<?php } ?> " 吃 


表单 处 理 页 savereply.php 将 表单 提交 的 数据 存储 到 指定 的 数据 库 中 ， 其 实现 的 方法 与 论坛 发 布 中 
的 表单 处 理 技术 是 相同 的 ， 有 关 该 技术 的 详细 讲解 请 参考 2.11.5 节 ， 这 里 不 再 袭 述 。 


2.11.7 单元 测试 
在 测试 网 上 社区 的 论坛 模块 时 , 发 现 发 帖 和 回帖 时 上 传 的 图 片 不 能 够 正常 显示 。 运行 结果 如 图 2.43 


所 示 。 分 析 错 误 原因 : 主要 有 两 个 方面 ， 一 是 图 片 没 有 上 传 到 指定 的 服务 器 文件 夹 下 ， 二 是 上 传 成 功 
后 ， 没 能 正确 地 读 取 数 据 库 中 指定 图 片 的 路 径 。 


为 当前 位 置 ?》 社 区 论坛 


Oru us 


发 贴 人 ;: TSofk 
| 三 便 箱 钨 ciea 号 Iz 直 同 回 
| 用 户 晤 别 : 车 通 会 员 Wu 
发 贴 台 答 : 


| 注册 时 间 ; 2007-11-18 08-42:43 nD CTD 


图 2.43 程序 运行 错误 结果 图 
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首先 ， 检 测 第 1 种 情况 ， 没 有 出 现 问题 ， 图 片 可 以 上 传 到 指定 的 文件 夹 下 ， 并 且 图 片 的 路 径 也 可 
以 存储 到 指定 的 数据 表 中 ， 从 而 排除 了 第 一 种 情况 的 可 能 。 

然后 ,看 第 2 种 情况 ， 查 看 获取 的 图 片 路 径 是 否 正确 。 发 现在 读 取 数据 库 中 图 片 路 径 的 代码 段 中 ， 
使 用 了 错误 的 字段 名 称 ， 数 据 库 中 图 片 路 径 存储 使 用 的 字段 名 是 photo， 而 在 程序 代码 段 中 使 用 的 却 是 
photos。 错 误 代 码 如 下 : 


<2?php 
if($infob[photo]!=""){ // 判 断 是 否 存在 图 片 
Sphotos=substr($infob[photos].2,70): // 获 取 图 片 存储 的 路 径 
echo (stripslashes($infob["content"])): // 输 出 图 片 的 内 容 
echo "<img src=\"$photos\">"; // 输 出 图 片 
}else{ 
echo (stripslashes($infob["content"])): // 输 出 帖子 内 容 
?> 


将 代码 段 中 的 字段 名 进行 修改 ， 然 后 重新 运行 程序 ， 图 片 正 常 显示 。 


2.12 后 台 首页 设计 


作为 一 个 完整 的 网 上 社区 系统 ， 要 想 能 够 及 时 地 对 网 站 进行 管理 和 维护 ， 必 须 具 有 一 个 强大 的 后 


台 管 理 系统 


2.12.1 后 台 首页 概述 


网 上 社 
包括 : 


站 


办 办 办 办 欠 罗 


下 面 看 


， 对 网 上 社区 系统 中 数据 进行 更 新 和 维护 。 


区 系统 的 后 台 管理 采用 的 是 一 种 简单 的 框架 结构 ， 通 过 switch 语句 来 实现 。 其 具体 内 容 


软件 试用 管理 : 包括 软件 适用 产品 的 添加 和 删除 。 

编程 词典 管理 : 包括 编程 词典 版 本 的 添加 、 删 除 和 编程 词典 内 容 的 添加 和 删除 。 
在 线 订 购 管理 : 主要 用 于 管理 用 户 提交 的 订单 。 

软件 升级 管理 : 包括 升级 包 的 添加 、 删 除 和 序列 号 的 添加 和 删除 。 


内 公告 管理 : 主要 用 于 添加 和 删除 站 内 公告 。 


技术 支持 管理 : 主要 用 于 添加 常见 问题 和 删除 常见 问题 ， 以 及 对 客户 反馈 信息 进行 管理 。 


一 下 本 案例 中 提供 的 后 台 首 页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\02\bcty365\admin\ 


index.php， 如 图 2.44 所 示 。 


2.12.2 后台 首页 技术 分 析 


网 上 社 


区 后 台 首页 的 设计 主要 应 用 switch 语句 和 include 包含 语句 来 实现 ， 其 实现 的 原理 是 : 应 用 


switch 语句 ， 根 据 超 链 接 中 传递 的 变量 值 进行 判断 ， 根 据 不 同 的 变量 值 应 用 include 包含 语句 调用 不 同 
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的 子 文件 。 该 技术 的 实现 流程 如 图 2.45 所 示 。 


版 本 名 各 | 
二 加 蝙 程 同 奥 开发 版 200T-12-08 11:52:34 EJ 
[Le 标 人 版 200r-12-06 1t:s2:0 | 部 

田 技术 支持 管理 简化 版 2007-12-08 11:51:4B | 地 

日 软件 试用 管理 专业 质 200r-l206 11:557 | 钊 


客户 服务 中 心 热 绩 : 0431-84978981 Copyrizht 2007 (化) 言 林 省 明日 科技 有 限 公司 


2.44 BCTY365 网 上 社区 系统 后 台 首 页 


入 [应 用 include 根 据 不 同 的 | 
值 输出 不 同 的 子 文件 


图 2.45 网 上 社区 后 台 首页 设计 流程 
为 能 够 更 好 地 理解 这 个 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 : 


switch( expr ){ //expr 条 件 为 变量 名 称 
case exprl: l/case 后 的 expr1 为 变量 的 值 
statementl: // 冒 号 ":" 后 的 是 符合 该 条 件 时 要 执行 的 部 分 
break : /应 用 break 来 跳 离 循环 体 
Case eXpI2 : 
statement2 : 
break : 
default: 


> 
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statementN; 
break; 


} 


参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变量 的 名 称 ; 参数 exprl 放置 于 case 语句 之 后 ， 是 
要 与 条 件 变量 expr 进行 匹配 的 值 中 的 一 个 ; statementl 是 在 参数 exprl 的 值 与 条 件 变量 expr 的 值 相 匹 
配 时 执行 的 代码 ; break 语句 实现 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 
跳出 循环 体 ，default 是 case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 
语句 。 

通过 switch 和 include 语句 来 实现 后 台 管 理 功能 的 设计 是 一 个 很 好 的 方法 ,不 但 实现 过 程 简单 ， 而 
且 操 作 也 非常 灵活 。 其 关键 代码 如 下 : 

例 程 19 ”代码 位 置 光盘 \TMW2\bcty365\adminvwzdh php 


<?php 
switch($htgD) { // 根 据 变量 提交 的 不 同 值 
case "添加 编程 词典 版 本 ": // 判 断 与 变量 提交 的 值 是 否 相 同 
include("addbb.php"): // 如 果 值 相 同 ， 则 调用 指定 的 文件 
break: 1/ 并且 跳出 本 次 循环 


case "编辑 编程 词典 版 本 ": 
include("editbd.php"): 
break: 


a // 部 分 代码 省 略 
case "" // 当 变量 的 值 为 空 时 
include("edittell.php"): // 调 用 该 文件 


break; 
} 


?> 
2.12.3 后台 首页 的 实现 过 程 


国 后台 首 页 使 用 的 数据 表 : tb_bb 

在 后 台 首 页 的 设计 过 程 中 ， 以 switch 循环 语句 为 基础 ， 架 设 整 个 后 台 管 理 功能 的 框架 结构 ， 充 分 
发 挥 include 包含 语句 的 作用 ， 调 用 不 同 的 文件 执行 不 同 的 管理 操作 ， 应 用 JavaScript 脚本 来 控制 栏目 
列表 的 输出 和 隐藏 。 

控制 栏目 列表 的 输出 和 隐藏 在 menu.php 文件 中 进行 ， 首 先 定义 一 个 函数 changeO 用 于 控制 表格 的 
输出 和 隐藏 ， 然 后 在 表格 中 应 用 onclick 事件 传递 不 同 的 值 到 自 定义 函数 change0， 最 后 根据 不 同 的 值 
显示 不 同 的 内 容 。 其 关键 代码 如 下 : 

例 程 20 ”代码 位 置 : de fn 

<script language="javascrip 


7 过 及 着 < 入 机 文 东 棋 的 伸展 和 收编 
function change(x.y){ /定义 一 个 函数 
if(x.style.display—"none"){ // 判 断 当 样 式 的 值 为 "none" 时 
x.style.display=""; // 判 断 当 样 式 的 值 为 "none" 时 ， 输 出 样式 的 值 为 空 
else if(x.style.display—""){ // 判 断 当 样 式 的 值 为 空 时 


17 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


x.style.display="none": // 判 断 当 样 式 的 值 为 空 时 ， 输 出 样式 的 值 为 "none” 
y.background="images/bg_16 11.jpe": // 输 出 背景 图 片 
} 
; 
</script> 


<table width="175" height="28" border="0" align="center" cellpaddine="0" cellspacine="4" 
onclick="change(tzl,img tz1)" style="cursor:hand"> 
<t> 
<td background="images/bg 16_11.jpg" id="img tzl" class="a4"><div align="left"><img 
src="images/bg_16 21.jpe">&nbsp: 编 程 词典 管理 </div></td> 
</> 
</table> 
<table name="tz1" id="tz1" width="170" height="40" border="0" align="center" cellpadding="0" cellspacing="0" 


<2php 
// 根 据 变量 的 值 选择 执行 的 内 容 ， 当 变量 的 值 不 为 真 时 ， 隐 藏 该 表格 
壕 !($_GET[htgl]j 一 "添加 编程 词典 版 本 ”||$_GET[htgl] 一 "编辑 编程 词典 版 本 ”||5_GET[htgl] 一 "添加 编程 词典 " 
|$_GET[htg1] 一 "编辑 编程 词典 " )){ 
?> 


style="display:none" 
Sphp > 
2 


<tr> 
<td width="40" height="24" background="images/bg_16 16jpe">&nbsp:</td> 
<td width="114" background="images/bg 16 _16.jpe"><div align="left"><a href="default.php?htgl= 添 加 编程 词典 
版 本 "> 添加 编程 词典 版 本 </a></div></td> 
</tr> 
</table> 


[| 说 明 : 这 里 给 出 的 只 是 后 台 首 页 实现 过 程 中 的 主要 代码 ， 详 细 代 码 可 参考 本 书 的 光盘 \TM\02\bcty365 
\admin\ 文 件 夹 下 的 相关 文件 。 


2.13 ”编程 词典 管理 模块 设计 
本 模块 的 功能 是 对 网 站 中 的 编程 词典 进行 管理 ， 包 括 添加 编程 词典 版 本 、 编 辑 编程 词典 的 版 本 、 
添加 编程 词典 和 编辑 编程 词典 。 
2.13.1 编程 词典 管理 模块 概述 
本 模块 的 主要 功能 是 管理 网 站 中 在 线 出 售 的 编程 词典 软件 ， 实 现 对 编程 词典 软件 及 时 的 更 新 和 维 
护 ， 其 管理 的 内 容 主要 包括 添加 和 编辑 编程 词典 的 版 本 ， 添 加 和 编辑 编程 词典 的 详细 信息 。 在 添加 编 
程 词典 时 ， 包 括 名 称 、 版 权 、 图 片 、 类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 ;编辑 编程 词典 包括 版 本 、 


价格 、 简 介 、 功 能 和 服务 ， 其 中 每 一 个 编程 词典 软件 只 可 以 编辑 一 次 ， 不 可 以 进行 重复 编辑 ， 如 果 要 
重新 编辑 ， 就 必须 将 已 经 编辑 过 的 信息 删除 。 
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2.13.2 ”编程 词典 管理 模块 技术 分 析 


在 编程 词典 管理 模块 中 ， 应 用 到 图 片上 传 技术 ， 通 过 该 技术 将 编程 词典 的 界面 效果 上 传 到 服务 器 
的 指定 文件 夹 下 。 该 技术 主要 通过 move_uploaded_file0) 函 数 来 实现 ， 其 中 还 应 用 到 is_dir0、mkdir0 函 
数 ， 来 判断 指定 的 文件 夹 是 否 存 在 和 创建 文件 夹 ， 还 用 到 mt_randO、strstr0 函 数 和 $_FILES[] 全 局 变量 。 
为 了 帮助 读者 更 好 地 理解 和 掌握 图 片上 传 处 理 技术 ， 这 里 以 编程 词典 模块 中 的 savebccd.php 文件 为 例 
进行 讲解 。 

首先 应 用 is_dir0 函 数 判断 在 服务 器 中 是 否 存 在 指定 的 文件 夹 ， 如 果 不 存在 ， 则 应 用 mkdir0 函 数 创 
建 一 个 新 的 文件 夹 。 

然后 应 用 $_FILES[] 全 局 变量 获取 图 片 名 ， 应 用 strstr0) 函 数 获取 图 片 文件 的 后 缀 名 ， 为 避免 出 现 同 
名 文件 覆盖 ， 这 里 应 用 系统 的 当前 时 间 和 mtrandO) 函 数 获取 的 一 个 7 位 随机 数字 作为 图 片 的 名 称 。 

最 后 确定 图 片 在 服务 器 中 存储 的 路 径 ， 将 图 片上 传 到 指定 的 文件 夹 下 。 而 数据 库 中 存储 的 数据 是 
图 片 在 服务 器 中 的 路 径 ， 当 需要 输出 图 片 时 ， 只 需要 获取 到 数据 库 中 图 片 的 路 径 即 可 。savebccd.php 
文件 的 代码 如 下 : 

例 程 21 代码 位 置 : 光盘 \TM\02\bcty365\admin\savebccd.php 

<?php include_once("../conn/conn.php"): // 连 接 数 据 库 

$bccdname=$_POST[bccdname]: // 获 取 POST 方 法 提交 的 值 

$owner=$_ POST[owner]: 

S$typeid=$_ POST[typeid]: 

$content=$_ POST[content]: 

$samepart=$_POST[samepart]; 


$addtime=date("Y-m-j H:i:s"): // 获 取 当 前 时 间 

@ ifis_dir("/bccdimages") 一 包 lse){ // 判 断 指定 的 文件 是 否 存 在 

@ mkdir("./bccdimages"): // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
b 

Slink=date("YmjHis"): // 获 取 当 前 时 间 


// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 
// 其 中 使 用 3_FILES 获 取 表单 中 真实 的 名 称 ， 使 用 strstt 函 数 获取 文件 的 后 级 
© $path=$link.mt rand(1000000.9999999).strstr($ FILES["imageaddress"]["name"],"."): 


Saddress="./bccdimages/".$path: /定义 文件 上 传 的 路 径 

@ move_uploaded file($ FILES["imageaddress"]["tmp_name"].Saddress): /将 文件 上 传 到 指定 的 文件 中 
Simageaddress="./admin/bccdimages/".$path: // 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysql query("insert into tb_bccd(bccdname.owner.typeid.content.samepart,imageaddress,addtime) 
values('$bccdname'.'$owner'.'$typeid'.'$content'.'$samepart'.'$imageaddress'.'$addtime")".$conn): 
这 $query 一 true){ 
echo "<script>alert( 编 程 词典 添加 成 功 ! ):history.backO:</script>": 
}else{ 
echo "<script>alert( 编 程 词典 添加 失败 ! '):history.back0:</script>"; 
} 


?> 


uy 
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< 人 代码 贴 二 
@ is dir0: 判断 指定 的 文件 夹 是 否 存 在 ， 如 果 存 在 ， 则 返回 true; 否则 返回 false。 
@ mkdir0: 创建 一 个 新 的 文件 夹 。 
四 mt rand0): 根 据 提供 的 参数 min 和 max 生 成 随机 数 ,如 果 没 有 提供 可 选 参数 min 和 max, 则 返回 0 到 RAND MAX 


之 间 的 伪 随机 数 。 

strstr0: 获取 一 个 指定 字符 串 在 另 一 个 字符 囊 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 囊 。 如 果 执行 成 功 ， 则 返回 剩 
余 字 符 串 (存在 相 匹配 的 字符 ) ; 如 果 没 有 找到 相 匹配 的 字符 ， 则 返回 false。 

$_FILES[]: 全 局 变量 ,获取 所 有 上 传 文件 的 信息 。 该 全 局 变量 还 可 以 获取 到 其 他 的 值 ， 其 中 $_FILES['imageaddress'] 
[mame'] 获 取 的 是 客户 端 机 器 文件 的 原名 称 ; $_FILES['imageaddress']['size'] 获 取 已 上 传 文件 的 大 小 , 单位 为 字 节 ; $_FILES 
[imageaddress']['tmp_name'] 获 取 文件 被 上 传 后 在 服务 端 储存 的 临时 文件 名 ; $_FILES['imageaddress']['error] 获 取 和 该 文件 
上 传 相关 的 错误 代码 。 

@ move uploaded file0: 应 用 POST 方法 实现 文件 的 上 传 ， 参数 filename 指定 要 上 传 的 文件 地 址 ; 参数 destination 
指定 文件 上 传 到 服务 器 后 的 存储 目录 及 名 称 。 


全 注意 : 应 用 POST 方法 上 传 图 片 文件 时 ， 应 当 在 上 传 表单 的 <form> 标 记 中 添加 以 下 内 容 enctype= 


"multipart/form-data"。 
2.13.3 ”添加 编程 词典 的 实现 过 程 


国 。 添加 编程 词典 使 用 的 数据 表 : tb bccd、tb type 
添加 编程 词典 的 功能 是 向 数据 库 中 添加 编程 词典 的 详细 信息 ， 包 括 编程 词典 的 名 称 、 版 权 、 图 片 、 
类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 。 其 运行 结果 如 图 2.46 所 示 。 


编程 间 典 名 称 : PE 编程 司 奥 
版 权 所 有 : 寺 林 省 明日 科技 有 限 公司 


图 片 路 径 : C:\Documents end Settines\| 


所 属 类 别 : T PEP 本 类 | 


A 为 广大 的 FIP 爱好 者 提供 一 国 | 
他 o 


内 容 漳 介 ; 
悦 
陡 对 不 同 的 版 本 , 蕊 中 涉及 到 不 同 堆 夏 程 度 的 知识 豆 
不 同 版 本 相同 点 : 
加 


EE 
图 2.46 添加 编程 词典 模块 的 运行 结果 


添加 编程 词典 信息 模块 主要 通过 addbccd.php 和 savebccd.php 文件 来 完成 ， 其 中 在 addbccdphp 文 
件 中 主要 是 设计 表单 元 素 ， 而 savebccd.php 文件 主要 是 对 表单 中 提交 的 数据 进行 处 理 。addbccd.php 文 
件 中 使 用 的 表单 元 素 如 表 2.7 所 示 。 


sn 
es 


第 2 章 BCTY365 网 上 社区 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


表 2.7 添加 编程 词典 页 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
method="post" action="savebccd.php" onSubmit="return chkinput(this)" 


forml form 编程 词典 表单 
enctype="multipart/form-data"> 
bccdname text size="25" class="txt grey" 编程 词典 名 称 
Owner text " size="25" class="txt_grey" 版 权 所 有 者 
imageaddress file size="25" class="txt_grey" 界面 图 片 
<?php include once("../conn/conn.php"): 
S$sql=mysql_query("select * from tb_type order by createtime desc",$conn); 
Sinfo=mysql fetch array($sq]): 
、 if($info 一 false){ echo "<option > 暂 无 类 别 </option>"; 选择 编程 词典 
typeid select i 
Jelse{ 的 版 本 


dof 
echo "<option value=".$info[id].">".Sinfo[typename]."</option>": } 
whbile(S$info=mysql_fetch_array($sqD):} ?> 


content textarea 内 容 简 介 
samepart textarea rows="10" cols="65" class="textarea"> 不 同 版 本 的 特点 
Submit submit Value=" 添 加 " class="btn_grey" 提交 表单 


savebccd.php 文件 实现 对 表单 中 提交 的 数据 进行 处 理 ， 首 先 通过 $_POST 获取 表单 中 提交 的 数据 ， 
然后 判断 指定 的 文件 夹 是 否 存在 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 关键 代码 如 下 : 


例 程 22 ”代码 位 置 : 光盘 \TM\02\bcty365\admin\savebccd.php 

<?php include_once("../conn/conn.php"): // 连 接 数 据 库 
$bccdname=$_POST[bccdname]: // 获 取 POST 方 法 提交 的 值 
$owner=$_ POST[owner]; 

$typeid=$_POST[typeid]: 

$content=$_POST[content]: 

$samepart=$_POST[samepart]: 


Saddtime=date("Y-m-j H:i:s"): // 获 取 当 前 时 间 

@ ifis dir(".bccdimages")=—false){ // 判 断 指定 的 文件 是 否 存在 

© mkdir("./bccdimages"): // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
} 
Slink=date("YmjHis"): // 获 取 当 前 时 间 


// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 ， 其 中 使 用 $_FILES 获 取 表 单 中 真实 的 名 称 ， 使 用 
strstt 函 数 获 取 文 件 的 后 缀 
@ spath=$link.mt rand(1000000.9999999).strstr($_ FILES["imageaddress"]["name"]."."): 


$address="./bccdimages/".$path: // 定 义 文件 上 传 的 路 径 

move_uploaded file($ FILES["imageaddress"]["tmp_name"],$address): // 将 文件 上 传 到 指定 的 文件 中 
Simageaddress="./admin/bccdimages/".$path: // 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 

/将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysql query("insert into tb_bccd(bccdname.owner.typeid.contentsamepartimageaddress.addtime) 
values('$bccdname','$owner'.'$typeid','$content'.'$samepart'.'$imageaddress'.'$addtime")". $conn); 
if($query—true){ 
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echo "<script>alert( 编 程 词典 添加 成 功 !"):history.back0:</script>"; 
}else{ 

echo "<script>alert(' 编 程 词典 添加 失败 !"):history.backQ 〇 :</script>"; 
bh 


?> 
外 代码 贴 二 

@ is dir0: 判断 指定 的 文件 是 否 存 在 。 

@ mkdir0: 创建 一 个 新 的 文件 夹 。 

@ mt rand0: 获取 随机 数字 。 

strstr0: 函数 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字 符 串 。 如 果 执 行 成 功 ， 则 返 
回 剩余 字符 串 〈 存 在 相 匹配 的 字符 ) ; 如 果 没有 找到 相 匹 配 的 字符 ， 则 返回 false。 


2.13.4 ”编辑 编程 词典 的 实现 过 程 
国 ”编辑 编程 词典 使 用 的 数据 表 : tb bccd、tb bb、tb bbqb 


在 完成 对 编程 词典 信息 的 添加 后 ， 接 下 来 就 可 以 对 编程 词典 的 版 本 信息 进行 编辑 ， 主 要 添加 版 本 
信息 、 价 格 、 简 介 、 功 能 和 推出 的 服务 。 该 模块 的 运行 结果 如 图 2.47 所 示 。 


蝙 程 词 典 名 称 添加 时 间 添加 版 本 信息 贡 
VB 编程 词典 2007-12-13 11:44:58 应 加 六 
VB 编程 词典 2007-12-02 10:29:42 添加 全 
版 本 名 称 价格 (元 ) | 内 容 简介 功能 服务 删除 
编程 词典 名 称 : VE 护 程 司 奥 
版 本 名 称 : [而 5 系 ”到 
价格 : Pooo 
加 本 全 面 司 
be 
ml 
风能 强大 司 
De 
| 
服务 弄 全 | 
RA 
加 
于 


图 2.47 编辑 编程 词典 模块 的 运行 结果 
该 功能 的 实现 同样 通过 两 个 文件 , 一 个 是 提交 表单 的 文件 editbccd.php 和 一 个 是 处 理 表单 提交 数据 
的 文件 sacvbccdbb.php。 提 交 表 单 文件 editbccd.php 中 使 用 的 表单 元 素 如 表 2.8 所 示 。 
表 2.8 编辑 编程 词典 页 中 使 用 的 重要 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含义 


method="post" action="savebccdbb.php" onSubmit="return 
ER 编辑 编程 词典 表单 


forml form 
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续 表 
称 | 元 素 类 型 重要 属性 含义 


bccdname text 


Class="txt grey" disabled= "disabled" value=" 

<?php 

$sql4=mysql query("select bccdname fiom tb_bccd where 编程 词典 名 称 , 这 里 设置 
id=".$_GET[bccdid]."",$conn): 了 文本 的 只 读 属性 

Sinfo4=mysql fetch array($sql4):; 

echo unhtml($info4[bccdname]): ?> 


bccdid hidden value="<?php echo $_ GET[bccdid]:?> 编程 词典 的 ID 


<?php $sql3=mysql query("select * from tb_bb order by 
createtime desc ",$conn):; 

Sinfo3=mysql fetch array($sql3); 

这 $info3 一 false){ echo "<option> 暂 无 版 本 信息 </option>"; 


bbid select }else{ 选择 编程 词典 的 版 本 


do{f ?> 
<option value="<?php echo $info3[id]:?>"><?php echo 
unhtml($info3[bbname]):?></option> 


<?php }while($info3=mysql_fetch_array($sql3)): }2> 


Submit Submit value=" 添 加 " class="btn_grey" 提交 表单 


sacvbccdbb .php 文件 对 表单 提交 的 数据 进行 处 理 , 首先 获取 表单 中 提交 的 数据 , 然后 判断 指定 的 版 


本 是 否 已 经 被 添加 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 代码 如 下 : 


例 程 23 ”代码 位 置 光盘 \TM\02\bcty365\admin\savebccdbb.php 


<?php 
$bccdid=$_ POST[bccdid]; // 获 取 表 单 中 提交 的 数据 
Sbbid=$_POST[bbid]: // 获 取 编 程 词典 id 
Sprice=$_POST[price]: // 获 取 编 程 词典 单价 
Scontent=$_POST[content]: // 获 取 编 程 词典 内 容 
$gn=$_POST[gn]; /获取 编程 词典 功能 
S$fw=$_ POST[fw]: // 获 取 编 程 词典 服务 
include_once("../conn/conn.php"): // 连 接 数 据 库 文件 
// 判 断 提交 的 编程 词典 是 否 已 经 被 添加 
$sql=mysql_query("select id from tb_bbqb where bccdid=".$bccdid."".$conn): 
Sinfo=mysql_fetch_array($sql): /检索 指定 编程 词典 的 id 
这 Sinfo!=false){ /如 果 检 索 值 为 值 ， 则 弹出 提示 
echo "<script>alert( 该 版 编程 词典 已 经 添加 !):history.backO:</script>": 
exit: } 


/ 率 半 六 冰 闪 六 当当 六 六 六 六 素 兴 六 这 六 率 床 闵 六 六 六 六 率 率 这 六 计 六 六 这 六 率 六 站 率 六 站 让 六 率 率 六 率 六 守 六 六 六 亲 订 订 半 让 订 计 率 床 六 来 二 六 / 


$query=mysql_query("insert into tb_bbqb(bccdid.bbid.price.content.en.fw) 


values('$bccdid','$bbid','$price'.'Scontent','$egn','$fw")", Sconn): // 将 表单 中 提交 的 数据 存储 到 数据 库 中 
/更 新 编程 词典 的 价格 
$querys=mysql query("update tb_bccd set bbid=-'$bbid'.price= Sprice' where id=".$bccdid."™"): 
if($query==true and $querys—true){ // 如 果 添 加 和 更 新 操作 为 真 ， 则 弹出 提示 


echo "<script>alert( 版 本 信息 添加 成 功 !"):history.back:</script>": 
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jelse{ // 如 果 添 加 和 更 新 操作 为 候 ， 则 弹出 提示 
echo "<script>alert( 版 本 信息 添加 失败 !"):history.backO:</script>": 
} 


?> 


2.14 软件 升级 管理 模块 设计 


2.14.1 软件 升级 管理 模块 概述 


软件 升级 管理 模块 实现 对 软件 升级 包 的 管理 ， 其 具体 的 功能 包括 添加 升级 包 、 编 辑 升级 包 、 添 加 
序列 号 和 编辑 序列 号 。 软 件 升级 管理 模块 中 的 添加 升级 包 和 添加 序列 号 是 一 一 对 应 的 ， 其 中 根据 所 属 
的 类 别 和 版 本 来 确定 升级 包 对 应 的 序列 号 ,每 一 个 版 本 一 个 类 别 的 升级 包 对 应 一 个 序列 号 。 


2.14.2 ”软件 升级 管理 模块 技术 分 析 


在 软件 升级 包 管理 模块 中 ， 应 用 到 一 个 动态 输出 下 拉 列 表 框 中 值 的 技术 。 下 面 就 来 讲解 一 下 该 技 
术 是 如 何 实现 的 ， 在 讲解 该 技术 之 前 ， 先 来 了 解 下 拉 列 表 框 的 基本 结构 : 
<select name="select"><!-->name 指 定 该 下 拉 列 表 框 的 名 称 <!--> 
<!-->selected 设 置 下 拉 列 表 框 的 默认 值 ,默认 值 为 PHP<!--> 
<option selected="selected">PHP</option> 
<!-->value 指 定 的 "mysql" 是 下 拉 列 表 框 传递 的 值 "MYSQL "为 显示 的 内 容 <!--> 
<option value="mysql">MYSQL</option> 
</select> 


所 谓 动态 输出 下 拉 列 表 框 中 的 值 就 是 从 数据 库 中 读 取 数 据 , 将 获取 到 的 数据 输出 到 下 拉 列 表 框 中 ， 
而 不 是 直接 在 下 拉 列 表 框 中 设置 某 个 固定 的 值 。 这 里 以 软件 升级 管理 模块 addsjb.php 文件 中 的 所 属 类 
别 下 拉 列 表 框 为 例 进行 讲解 ， 其 中 设置 下 拉 列 表 框 的 名 称 为 typeid， 默 认 值 为 “请 选择 ”，value 的 值 
是 从 数据 库 中 获取 的 ID 值 ， 显 示 的 内 容 为 从 数据 库 中 获取 的 类 型 名 称 。 动 态 输 出 下 拉 列 表 框 中 的 值 使 
用 的 关键 代码 如 下 : 

例 程 24 ”代码 位 置 : 光盘 \TM\02\bcty365\ admin\wzdh.php 


到 设置 下 拉 列 表 框 的 名 称 为 typeid------------------------------------- > 
<select name="typeid" class="txt_grey"> 
<option value="" selected="selected"> 请 选择 </option> 


< -一 一 一 一 一 -一 一 -一 -设置 下 拉 列 表 框 的 名 称 为 ypeid-- 一 -一 一- 一 -一 -一 一 一- -> 
<?php 

include_once("../conn/conn.php"): // 连 接 数 据 库 

/从 数据 库 中 读 取 编程 词典 类 型 的 数据 


S$sql=mysql query("select * from tb_type order by createtime desc".Sconn): 
Sinfo=mysql fetch_array($sq]): 
if($info—false){ 

echo "<option > 暂 无 类 别 </option>":; 
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}else{ 


dof /应 用 do…while 循 环 语句 输出 类 型 的 ID 和 类 型 的 名 称 
echo "<option value=".$info[id].">".$info[typename]."</option>"; 
} 
while($info=mysql_fetch_array($sqD)); //do…while 循 环 语句 结束 
} 
> 
</select> 


下 拉 列 表 框 不 但 可 以 动态 输出 数据 库 中 某 个 字段 的 数据 ， 而 且 可 以 输出 数组 中 的 数据 。 下 面 就 实 
现 一 个 在 下 拉 列 表 框 中 动态 输出 数组 中 数据 的 功能 ， 首 先 创建 一 个 下 拉 列 表 框 ， 然 后 设置 下 拉 列 表 框 
的 值 ， 从 数组 中 读 取 数 据 ， 应 用 for 循环 语句 进行 输出 。 其 代码 如 下 : 


< -设置 下 拉 列 表 框 的 名 称 为 select-------------------------------------- -> 

<select name="select" size="1"> 

<?php 
S$string="ASP.PHP.JSP.NETDEL.VB.VC": /定义 一 个 字符 串 
Ssrtings=split(",",$string): // 对 字符 串 进行 分 割 
Scount=count(S$srtings): // 获 取 数 组 中 元 素 的 数量 
for($i=0;$i<$count:$i++) { /根据 数组 中 的 元 素 的 数量 进行 循环 输出 

Sresult=$srtings[$i]: // 定 义 变量 ， 获 取 数 组 中 指定 的 元 素 

echo "<option>$result</option>":} // 将 数组 中 的 元 素 输出 到 下 拉 列 表 框 中 

?> 

</select> 


动态 输出 数据 库 中 数据 到 下 拉 列 表 框 的 运行 结果 如 图 2.48 所 示 。 
编程 词典 名 称 : [Pf 编程 语 奥 

版 权 所 有 尖 林 省 明日 科技 有 限 公司 

图 片 路 径 : [CDocments and Settines\| DRO 


所 属 类 别 : 请 选择 了 


内 容 简 介 : 


『 Visual C++ 】 类 
『 Power puilder 】 类 


ne 


! 这 司 
图 2.48 动态 输出 数据 库 中 数据 到 下 拉 列 表 框 


2.14.3 ”软件 升级 包 上 传 的 实现 过 程 


国 ”软件 升级 包 上 传 使 用 的 数据 表 : tb bb、tb type、tb sjxz 

软件 升级 包 上 传 在 添加 升级 包 模 块 中 实现 ， 通 过 一 个 文件 域 文本 框 将 升级 包 提交 到 服务 器 中 指定 
的 文件 下 ， 并 且 将 该 文件 在 服务 器 中 的 路 径 存 储 到 数据 库 中 ， 便 于 在 前 台 实 现 对 软件 升级 包 的 下 载 。 
其 运行 结果 如 图 2.49 所 示 。 
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加 升级 包 
升 红包 名 称 : Pe 
所 属 类 别 : [LE 了 
版 本 名 称 : 标准 版 加 
下 载 由 让: EGR setter lm 
所 司 
说 明 及 简介 
二 
失实 | 二 S| 


2.49 ”软件 升级 包 上 传 的 运行 结果 


在 本 模块 中 通过 addsjb.php 文件 来 提交 升级 包 的 信息 , 通过 savesj.php 文件 来 对 表单 提交 的 数据 进 
行 处 理 。 其 中 在 将 升级 包 上 传 到 服务 器 的 指定 文件 夹 的 过 程 中 ， 主 要 应 用 的 是 move_uploaded file0 函 
数 。 在 savesj.php 文件 中 ， 首 先 获取 表单 提交 的 数据 ， 然 后 判断 服务 器 中 是 否 存在 指定 的 文件 ， 最 后 应 
用 move_uploaded_file0 函 数 将 升级 包 上 传 到 指定 的 文件 夹 下 ， 并 且 将 数据 存储 到 指定 的 数据 表 中 。 其 
程序 代码 如 下 : 

例 程 25 ”代码 位 置 : 光盘 \TM\02\bcty365\admin\savesj.php 


<?php 

Sname=$ POST[name]: // 获 取 表单 提交 的 数据 

$typeid=$_POST[typeid]: /获取 表单 提交 的 数据 

Scontent=$_POST[content]: // 获 取 表 单 提交 的 数据 

Saddtime=date("Y-m-j H:i:s"); // 定 义 时 间 变 量 

Sbbid=$_POST[bbid]; // 获 取 表 单 提交 的 数据 

iis_dir("./sjxz") 一 包 lse){ // 判 断 指定 的 文件 夹 是 否 存 在 
mkdir("./sjxz"): // 如 果 指 定 的 文件 夹 不 存在 ， 则 创建 一 个 指定 的 文件 夹 

} 

Slink=date("YmjHis"): // 获 取 一 个 时 间 

$path=$link.mt_rand(1000000,9999999).strstr($_FILES["address"]["name"],"."):// 重 新 设置 升级 包 名 称 

Saddress="./sjxz/".$path: // 设 置 升级 包 在 服务 器 中 存储 的 指定 路 径 

move_uploaded file($ FILES["address"]["tmp_name"].$address): // 将 升级 包 上 传 到 指定 的 路 径 下 

Saddress="./admin/sjxz/". $path: // 获 取 升 级 包 在 服务 器 中 的 存储 路 径 

include_once("../conn/conn.php"): // 连 接 数 据 库 文件 


// 将 上 传 的 数据 存储 到 数据 库 中 ， 这 里 将 升级 包 在 服务 器 中 的 路 径 存储 到 数据 库 中 
$query=mysql_query("insert into tb_sjxz(name,typeid.content,addtime,address,bbid) 
Values('$name','$typeid','$content','$addtime','$address','$bbid'")", $conn): 


if($query){ /如 果 添 加 操作 成 功 ， 则 弹出 提示 
echo "<script>alert( 升 级 包 添加 成 功 !"):history.backO0:</script>"; 
}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 


echo "<script>alert(' 升 级 包 添加 失败 !):history.backO:</script>": 


} 
?> 


2.14.4 ”软件 升级 包 删 除 的 实现 过 程 


国 ”软件 升级 包 删 除 使 用 的 数据 表 : tb_ bb、tb type、tb_sjxz 
软件 升级 包 删 除 的 实现 主要 根据 当前 数据 中 提供 的 ID， 执行 delete 删除 语句 ， 将 数据 表 中 相同 ID 
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的 数据 删除 。 其 运行 结果 如 图 2.50 所 示 。 
加 你 当前 的 位 置 : 代入 升 纪 
日 编程 词典 管理 
田 技术 支持 管理 
旦 款 件 试用 管理 __ 升级 包 版 本 | 。 升级 包 关 到 天 姐 包 各 和 更 新 时 间 Mp 
电 软件 升级 管理 | | Ve 统 闻 捧 3 0 升 妇 包 2007-12-02 14:32:51 


MN 个 条 0 个 第 1 页 / 共 1 页 1 
编 千 升 领 忆 es 
Er 


妨 各 序列 号 
田 在 线 订购 管理 
田 站 内 公告 管理 


客户 服 务 中 心 起 如 :091-04975981 copyritht 2007 5) 吉林 省 明日 科技 有 限 公司 
图 2.50 软件 升级 包 删 除 的 运行 结果 


该 功能 的 实现 主要 通过 editsjb.php 文件 和 deletesjb.php 文件 。 通 过 editsjb.php 文件 输出 数据 库 中 存 
储 的 有 关 升 级 包 的 信息 ， 以 分 页 的 形式 显示 ， 在 每 条 记录 的 最 后 设置 一 个 删除 链接 ， 通 过 脚本 来 调用 
deletesjb.php 文件 ， 根 据 变 量 中 的 ID 值 执行 删除 升级 包 的 操作 。 其 关键 代码 如 下 : 


例 程 26 ”代码 位 置 ， 光盘 \TM\02\bcty365\admin\deletesjb.php 

<?php 

$id=$_GET[id]: 

include_once("../conn/conn.php"): 

/| 执行 删除 操作 ， 将 数据 表 中 对 应 的 人 D 的 数据 删除 

这 mysql_query("delete from tb_sjxz where id=".$id."",$conn)){ 
echo "<script>alert(' 该 升级 包 删 除 成 功 !"):history.backQ0:</script>"; // 如 果 删 除 操作 成 功 ， 则 弹出 提示 

}else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 
echo "<script>alert(' 该 升级 包 删 除 失败 !"):history.backO0:</script>"; 


} 
?> 


/获取 变量 传递 的 ID 
/连接 数据 库 


2.15 在 Linux 系统 下 发 布 网 站 


在 Linux 下 发 布 基于 PHP 的 网 站 , 首先 需要 配置 PHP 的 运行 环境 , 其 次 需要 对 网 卡 参数 进行 设 定 。 
这 里 将 以 发 布 “BCTY365 网 上 社区 ”网 站 为 例 讲解 在 Linux 下 如 何 实现 网 站 的 发 布 。 假 设 已 经 申请 到 
表 2.9 所 示 的 网 络 参数 。 


表 2.9 申请 到 的 网 络 参数 


参数 值 
P 192.168.1.* 
Netmask 255.255.255:* 
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续 表 
参数 值 
Network 192.168.1.0 
Broadcast 192.168.1.* 
Gateway 192.168.1.* 
主机 名 Tsoft 
DNS 168.95.1* 


Linux 下 网 站 发 布 的 操作 步骤 如 下 : 
(1) 配置 PHP 的 运行 环境 ， 在 2.4 节 中 已 经 做 了 详细 介绍 ， 这 里 不 再 袭 述 。 
(2) 将 “BCTY365 网 上 社区 ”网 站 的 所 有 文件 复制 到 Apache 主 目录 下 。 
(3) 设置 主机 名 称 。 在 终端 窗口 中 输入 如 下 命令 编辑 /etc/sysconfig/network 文件 : 


Vi /etc/sysconfig/network 


将 该 文件 中 的 参数 NETWORKING 设置 为 yes， 表 示 启 动 网 络 ， 将 参数 HOSTNAME 设置 Tsoft， 
表示 设置 主机 名 为 Tsoft。 
(4) 设置 网 卡 参数 。 在 终端 窗口 中 输入 如 下 命令 编辑 文件 /etc/sysconfig/network-scripts/ifcfg-eth0: 


Vi /etc/sysconfig/network-scripts/ifcfe-ethO 
该 文件 的 相关 参数 设置 如 表 2.10 所 示 。 
表 2.10 设置 网 卡 的 相关 参数 


参数 说 了 明 
DEVICE=eth0 设置 网 卡 名 称 ， 要 与 这 fe-eth0 对 应 
ONBOOT=yes 指定 在 开机 时 启动 网 卡 
BOOTPROTO=static 设 定 启动 时 获取 人 P 的 方式 
IPADDR=192.168.1.* 设 定 服务 器 他 地 址 
NETMASK=255.255.255.* 设 定子 网 掩 码 
BROADCAST=192.168.1.* 设 定 同 网 段 的 广播 地 址 
GETWAY=192.168.1.* 设 定 网 卡 的 网 关 


(5) 设置 DNS 主机 的 全 。 在 终端 编辑 /etc/resolv.conf 文件 : 


Vi /etc/resolv.conf 


设置 参数 nameserver 的 值 为 168.95.1.*。 
(6) 重新 启动 网 络 设置 。 在 终端 窗口 中 输入 如 下 命令 : 


/etc/rc.d/inin.d/network restart 
ifdow eth0 
ifup eth0 
(7) 打开 浏览 器 ， 在 地 址 栏 中 输入 服务 器 的 IP 地 址 或 域名 ， 打 开 如 图 2.51 所 示 的 页 面 ， 则 说 明 
Linux 下 的 “BCTY365 网 上 社区 ”网 站 发 布 成 功 。 
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TITTTETTTETTTT7TIEITTIETTTRTT7 
立 - 多 . 了 hp /192 168 1 59/Tsofu/betya85/ 
a ee 


i 


See "| 


图 2.51 Linux 下 “BCTY365 网 上 社 


只 


站 运行 结果 
2.16 ”开发 技巧 与 难点 分 析 


2.16.1 管理 员 权 限 的 设置 


为 了 更 好 地 管理 和 维护 网 站 的 论坛 ， 针 对 论坛 设置 了 一 个 管理 员 ， 该 管理 员 不 在 后 台 进 行 操作 ， 
而 是 在 前 台 为 管理 员 设置 特殊 的 权限 ， 也 可 以 称 之 为 版 主 。 其 实现 的 原理 是 : 首先 在 数据 库 中 设置 不 
同 的 值 代表 不 同 的 权限 ，“0” 代 表 普 通 会 员 ，“1?” 代 表 管 理 员 ， 然 后 在 论坛 的 页 面 中 进行 判断 ， 当 
用 户 的 类 型 为 “1” 时 ， 不 但 具有 普通 会 员 的 权限 ， 而 且 具 有 删除 发 布 帖子 、 回 复 帖 子 和 顶 帖 的 权限 ; 
如 果 用 户 的 类 型 不 是 “1”， 则 不 具有 上 述 的 权限 ， 只 能 是 发 布 和 回复 帖子 。 管 理 员 和 普通 会 员 登 录 的 
页 面 效 果 是 不 同 的 ， 如 图 2.52 和 图 2.53 所 示 。 


CEITTTTT 


pe O me 


发 巾 人 ;lx 


信箱 坊 0ic4 时 本 地 址 起 证 划一 下 加 何 配置 PE 的 环境 
用 户 好 别 : 管理 角 
发 由 局 数 : 2 
注册 时 间 : 2007-12-01 13;15:25 ED 区 于 本 Des 7 
烙 Ori ae [reo 
同 复 主题 : PI 的 环境 是 方法 
有 同人 :lz 
信箱 忽 0ict 虹 本 地 址 。 8 内 罕 请 学 委 pr 玫 电 庄 系 纺 开 发 元 全 手 用 中 的 划 ] 重 
用 户 红 ; 管理 再 
发 由 总 数 : 2 


广 册 时 间 ; 2007~12-01 13:15:25 


图 2.52 管理 员 登 录 的 操作 页 面 
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哪 丢 的 计算 机 不 管 喇 动 了 ? 


发 贴 总数: 了 
广 册 时 间 : 2007-11-18 08:42:43 


崔 QOaoreeuna | 
宫 
EA: Tsoft DN Te sR 
信箱 疙 0ic4 蜡 TT 地 址 请 为 抬 解 决 一 下 
用 户 绿 别 ; 普通 会 员 | | 


图 2.53 普通 会 员 登 录 的 操作 页 面 


在 页 面 中 执行 的 判断 语句 判断 登录 用 户 的 类 型 ， 然 后 根据 类 型 判断 用 户 的 权限 。 其 程序 的 关键 代 
码 如 下 : 


<?php 
if($_SESSION["unc"]!=""){ // 浏 断 session 变 量 的 值 是 否 为 空 
/根据 session 变 量 的 值 获取 该 用 户 的 类 型 
$sqlu=mysql_ query("select usertype from tb_user where usernc=".$_SESSION["unc"]."".$conn); 
Sinfou=mysql fetch array($sqlu):; 
if($infou["usertype"]—1){ // 判 断 当 该 用 户 的 类 型 等 于 "1" 时 执行 下 面 的 内 容 
和 > 


<!--> 调 用 JavaScript 脚 本 ， 执 行 删除 发 布 帖子 的 操作 <!--> 
<img src="images/lt_15(10).jpg" onclick="javascript:if(window.confirm(' 您 确定 删除 该 帖 么 ? 
"=—=true) {window.location.href='bbs_delete.php?id=<?php echo $infob["id"]?>":}" style="cursor:hand"/> 
<?php 
} 
// 如 果 用 户 的 类 型 不 是 "1"， 则 不 执行 上 述 的 内 容 
) 


?> 


2.16.2 ”帖子 置顶 的 设置 


所 谓 帖子 置顶 就 是 将 某 个 指定 的 帖子 在 对 应 的 版 块 中 最 前 面 的 位 置 显示 ， 该 权限 只 有 管理 员 才 拥 
有 ， 普 通 会 员 不 具备 该 权限 。 其 实现 的 原理 为 : 

首先 ,在 数据 库 中 存储 发 布 帖子 信息 的 数据 表 中 设置 一 个 字段 top， 指 定 该 字段 为 数字 类 型 ， 其 默 
认 值 为 0。 

然后 ， 在 网 页 中 判断 登录 用 户 的 权限 ， 如 果 是 管理 员 则 具有 帖子 置 项 的 权限 ， 否 则 将 弹出 提示 对 
话 框 “对 不 起 ， 您 不 具备 该 操作 权限 ! ”。 

最 后 ， 如 果 是 管理 员 则 执行 settop.php 文件 ， 根 据 对 应 帖子 的 ID 查找 到 发 布 帖子 信息 表 中 对 应 的 
数据 ， 更 新 该 条 数据 中 top 字段 的 值 为 1。 


判断 登录 用 户 权 限 使 用 的 代码 如 下 : 
<?php 
if($_SESSION["unc"]—""){ // 判 断 session 变 量 的 值 是 否 为 空 ， 如 果 为 空 则 执行 下 面 的 脚本 
echo "javascript:alert( 请 先 登录 本 站 ， 然 后 进行 此 操作 !"):window.location href~'index.php':": 
}else{ 
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// 如 果 不 为 空 则 执行 下 面 的 内 容 ， 判 断 登录 用 户 的 权限 

$sqlu=mysql query("select usertype from tb_user where usernc=".$_SESSION["unc"]."".$conn): 

Sinfou=mysql fetch array($sqlu):; 

这 S$infou["usertype"] 一 1){ /如 果 登 录用 户 的 类 型 是 "1"， 则 说 明 是 管理 员 ， 则 执行 下 面 的 内 容 
echo "javascript:window.location.href='settop.php?id=".$infob["id"].""™"; 


}else{ 
// 如 果 登 录用 户 不 是 管理 员 ， 则 执行 下 面 的 内 容 
echo "javascript:alert( 对 不 起 ， 您 不 具备 该 操作 权限 1):"; 
} 
9 


?> 

实现 帖子 置顶 是 通过 settop.php 文件 来 完成 的 , 在 该 文件 中 , 根据 变量 提交 的 值 获取 到 发 布 帖子 信 
息 表 中 对 应 的 数据 ， 更 新 该 条 数据 中 字段 top 的 值 ， 并 且 对 该 字段 的 值 进行 判断 。 如 果 字 段 top 的 值 为 
1， 则 说 明 该 帖 已 经 置 项 ， 此 时 将 字段 的 值 更 新 为 0， 即 取消 置 项 ， 如 果 字 段 top 的 值 为 0， 则 说 明 该 
帖 没 有 进行 置顶 ， 此 时 将 字段 的 值 更 新 为 1， 即 置顶 该 帖 。settop.php 文件 的 程序 代码 如 下 : 

<?php include once("conn/conn.php"): // 连 接 数 据 库 文件 

/根据 获取 的 ID 值 ， 从 数据 表 中 读 取 到 对 应 的 数据 

$sql=mysql query("select top from tb_bbs where id=".$_GET["id"]."",$conn): 


Sinfo=mysql fetch_array($sq)l):; 
这 $info[top] 一 1){ // 判 断 对 应 数据 记录 中 的 字段 top 的 值 ， 如 果 字 段 top 的 值 为 "1"， 则 执行 下 面 的 内 容 


mysql_query("update tb_bbs set top=0 where id=".$_GET["id"]."",$conn); // 更 新 字段 top 的 值 为 0 
}elseif($infoftop]==0){ // 如 果 对 应 数据 记录 中 的 字段 top 的 值 为 "0"， 则 执行 下 面 的 内 容 
mysql_query("update tb_bbs set top=1 where id=".$_GET["id"]."",$conn): // 更 新 字段 top 的 值 为 1 
} 
echo "<script>alert( 置 上 设置 成 功 !"");history.backQ;</script>"; 
?> 


2.17 在 线 支付 技术 专题 


所 谓 在 线 支付 就 是 客户 端 (金融 机 构 需 客户 端 安装 由 金融 机 构 签发 之 数字 证 书 ， 信 用 卡 免 安装 ) 
将 支付 信息 加 密 后 通过 互联 网 传送 到 支付 网 关 〈 支 付 网 关 是 解决 网 络 上 安全 支付 问题 的 交易 平台 ， 位 
于 互联 网 和 传统 的 金融 机 构 内 部 网 之 间 ， 其 主要 作用 是 将 互联 网 和 金融 网 络 安全 的 连接 起 来 ， 将 不 安 
全 的 网 上 交易 信息 传 给 安全 的 金融 网 络 ， 起 到 隔离 和 保护 金融 网 络 的 作用 ) ， 同 时 金融 机 构 网 上 支付 
系统 反馈 有 关 支付 信息 ， 客 户 确 认 无 误 后 进行 支付 确定 ， 支 付 网 关 负 责 商户 网 上 交易 资金 的 清算 ， 并 
根据 商户 提供 的 开户 行 、 账 号 等 结账 信息 将 网 上 消费 款项 汇总 划 入 商户 账户 。 

BCTY365 网 上 社区 的 在 线 支 付 是 与 中 国 工商 银行 合作 来 共同 完成 的 。BCTY365 网 上 社区 的 在 线 
支付 操作 步骤 如 下 : 

(1) 登录 网 上 社区 ， 如 图 2.54 所 示 。 

(2) 购买 商品 。 在 本 页 中 ， 不 但 可 以 购买 商品 ， 还 可 以 查看 商品 的 详细 信息 和 购物 车 中 的 商品 信 
息 ， 如 图 2.55 所 示 。 
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(3) 进入 购物 车 操作 页 面 。 在 该 页 面 中 ， 可 以 修改 购物 数量 ， 删 除 指定 商品 ， 清 空 购物 车 ， 继 续 
购物 和 统计 购买 商品 的 金额 ， 也 可 以 单 击 “ 结 算 ” 按 钮 进入 到 商品 结算 页 面 ， 如 图 2.56 所 示 。 


| 
时 Ea 

至 上 所 展 版 本 ; 考 业 版 

| SS 价格 : 20000 元 

:| = bE) 版 。 可 :吉林 省 明日 科技 有 限 公司 


到 
3 价格 : 2&000.00 元 
= 是 。 。 版 祝 ; 吉林 省 明日 科技 有 限 公司 商号 次 | 单价 (元 | 效 是 (不 ) Ea 
ED Er Gs A Co 
一 ~ 人 | | < 多 绕 购 买 清空 购物 车 商品 全 额 SSit: 2,000.00 元 。 2 放 
图 2.55 ”购买 商品 操作 页 面 图 2.56 购物 车 操作 页 面 


(4) 进入 到 购物 结算 页 面 ， 填 写 收 货 人 的 详细 信息 ， 确 认 后 提交 该 数据 ， 如 图 2.57 所 示 。 
“ 收 货 人 信息 * 请 务必 正确 填写 悠 的 个 人 详细 信息 + 


收 人 人 : ee 
世 别 : 三 更 司 


tt: 区 
邮政 编码 : [30000 
ms 码 : [i8381769 = 
Enail: [inaeisoftasine em 

* 语 务 必 有 正确 填写 雹 的 联系 地 址 和 地 篇 ， 以 确保 订单 和 货物 须 利 达到 9 


移动 电话 : 360433eee* 固定 电话 : 5451-84976961 


邮递 方 式 * 请 选择 送 货 方 式 4 


图 2.57 填写 收 货 人 的 详细 信息 
(5) 订单 确认 。 订 单 确认 以 后 ， 就 可 以 提交 订单 ， 准 备 进行 网 上 支付 ， 如 图 2.58 所 示 。 


og 


名 
[> 
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吉林 省 明日 科技 有 限 公司 -编程 司 风 订单 


订单 号 : 07121311451945 
收 货 人 : 潘 * 
移动 电话 : 1360433++t+ 
联系 地 址 : 长 春 市 


商品 名 称 
他 编程 局 奥 


2,000.00 1 


2007-12-13 11:54:39 


邮编 : 130000 
固定 电话 : 0431-84978981 


商品 总 计 : 2,000.00 元 
邮费 : 240.10 元 
你 需要 支付 的 人 金额 总 计 为 : 2,240 10 元 


小 计 ( 元 》 


2,000.00 


图 2.58 订单 确认 


(6) 进行 网 上 支付 。 在 这 里 可 以 选择 工行 网 上 支付 ， 也 可 以 选择 取消 该 订单 ， 如 图 2.59 所 示 。 


订单 号 : 07121311451945 


需 支 付 全 额 : 2240.1 元 


图 2.59 执行 网 上 支付 


接 下 来 的 操作 在 工行 B2C 支付 页 面 上 进行 ,首先 网 上 社区 按照 工商 银行 B2C 订单 数据 规范 形成 提 
交 数 据 ， 并 使 用 工商 银行 提供 的 API 和 商户 证 书 对 订单 数据 签名 ， 形 成 form 表单 返回 客户 浏览 器 ， 表 
单 action 地 址 指向 工商 银行 接收 商户 B2C 订单 信息 的 servlet; 然后 在 客户 确认 使 用 工行 网 上 支付 后 ， 
提交 此 表单 到 工商 银行 ， 最 后 工行 网 银 系统 接收 此 笔 B2C 订单 ， 对 订单 信息 和 商户 信息 进行 检查 ， 通 


过 检查 则 显示 工行 B2C 支付 页 面 。 


客户 通过 工行 B2C 支付 页 面 实现 网 上 支付 ， 商 户 查询 网 上 银行 的 账户 ， 如 果 货款 已 经 到 账 ， 则 根 


据 客户 指定 的 方式 将 货物 送 达 客户 手中 。 


上 述 内 容 就 是 网 上 社区 系统 的 在 线 支付 流程 ， 涉 及 到 工商 银行 的 操作 内 容 在 此 不 做 讲解 。 这 里 主 
要 讲解 一 下 如 何 将 订单 信息 提交 到 工商 银行 。 该 项 操作 主要 通过 shopping_tjdd.php 文件 来 实现 ， 首 先 
从 数据 库 中 读 取 订单 信息 ， 然 后 将 订单 信息 进行 输出 ， 最 后 创建 “取消 订购 ”和 “工行 网 上 支付 ”两 
个 超 连接 ， 通 过 JavaScript 脚本 来 调用 不 同 的 执行 文件 。 其 关键 代码 如 下 : 


<?php include once("conn/conn.php"): include once("top.php"): 
<!--> 省 略 了 部 分 代码 <!--> 

<?php 

Sddnumber=base64_decode($ GET["ddno"]): 

/获取 该 订单 的 金额 信息 

$sql=mysql_query("select * from tb dd where ddnumber=".$ddnumber. 
S$info=mysql fetch_array($sq]): 

$amount=$info["totalprice"]: 

Samount=str_replace("."."".number format($amount.2)): 
$amount=str_replace("."."".number format($amount.2)): 

?> 


/连接 数据 库 和 网 站 的 头 文件 ?> 


/对 获取 的 订单 编号 进行 base64 解 码 


.$conn): 


/修改 数字 的 输出 格式 
/修改 数字 的 输出 格式 


/省 略 部 分 HTML 代码 
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<table width="630" border="0" align="center" cellpaddine="0" cellspacine="0"> 
<tr><td width="159">&nbsp: 
<?php 
S$sql=mysql query("select totalprice from tb dd where ddnumber=".base64 decode($ GET["ddno"])."".$conn): 
Sinfo=mysql fetch array($sq)): 
echo "<font color=red><strong>".$info["totalprice"]."&nbsp: 元 </strong></font>"; 
> <td> 
</u> 
</table> 
多 /省 略 部 分 HTML 代 码 
<script language="javascript"> 
/打印 订单 
© function openprintwindow(x,y,Zz){ 
window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635,height="+( 
230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=n0o"); 


} 


</script> 
/省 略 部 分 HTML 代 码 
<table width="630" height="25" border="0" align="center" cellpadding="0" cellspacing= "0"> 
<tr> 
EE 二 > 


<td width="75"><img src="images/bg_14(14).jpg" width="69" height="20" style="cursor:hand" 
onclick="javascript:if(window.confirm(' 如 果 取 消 该 订单 ， 则 该 订单 将 被 删除 ， 您 需要 重新 购买 ! 
"==trme) {window.location.href='deletedd.php?ddno=<?php echo $_GET["ddno"]:?>": 


< width="125"><img src="images/bg_14(15).jpe" width="119" height_"20" 
onclick="javascript:window.location.href='ddform.php?orderid=<?php echo 
base64_decode($ GET["ddno"]);?>&amount=<?php echo $amount;?>&orderDate=<?php echo date("Ymdhis");?>";" 
style="cursor:hand"/></td> 
</u> 

</table> 

一 二 省 咯 了 部 分 代码 一 -一 一 一 一 一 > 

<?php ”include_once("bottom.php"): // 包 含 网 站 的 尾 文件 ?> 


< 代码 贴 十 

@ base64 decode0: PHP 实现 对 base64 编码 的 字符 进行 解码 。PHP 实现 字符 囊 的 base64 编码 通过 base64_encode() 
函数 。 

@ str_ replace0: 实现 字符 囊 的 替换 。 该 函数 的 语法 如 下 : 

mixed str_ replace ( mixed search. mixed replace, mixed subject . int &count ) 

str_replace() 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ， 参 数 &count 表示 替换 字符 串 执行 的 次 数 。 

@ openprintwindow 0: JavaScript 脚 本 中 自 定义 的 函数 ， 用 于 执行 订单 的 打印 操作 。 


有 关 在 线 支付 流程 中 的 其 他 操作 实现 方式 已 经 在 2.10 节 中 进行 了 详细 的 讲解 ， 这 里 不 再 歼 述 。 其 
具体 的 代码 读者 可 以 参考 本 书 的 光盘 中 的 光盘 \TM\02\bcty365\ 文 件 。 
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社 


2.18 本章 总 结 


本 章 从 项 目 开发 的 实际 角度 出 发 ， 以 某 科 技 公司 的 实际 需求 为 背景 ， 详 细 地 讲解 “BCTY365 网 上 


区 ”系统 的 开发 过 程 ， 其 中 以 系统 的 整体 开发 流程 为 主线 ， 重 点 介绍 技术 支持 、 在 线 订购 、 社 区 论 


坛 和 编程 词典 等 几 个 大 模块 的 实现 方法 ， 并 且 对 管理 员 权 限 设置 、 帖 子 置 顶 设置 和 在 线 支付 技术 作 技 
巧 和 专题 进行 讲解 ， 而 且 在 本 章 中 还 讲解 在 Linux 下 如 何 搭建 PHP 的 开发 环境 以 及 在 Linux 下 如 何 发 
布 网 站 。 
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第 章 


办 公 自 动 化 管理 系统 
(Apachet PHP+phpMyAdmin+MySQL 5.0 实现 ) 
(器 " 视频 讲解 : 1 小 时 40 分 钟 ) 


办 公 室 自动 化 的 英文 原 称 为 Office Automation，, 简称 为 OA， 办 公 自 动 化 系统 
指 实现 办 公 室 内 事务 性 业务 的 自动 化 。 

办 公 自 动 化 没有 明确 的 定义 , 最 普遍 的 说 法 是 凡是 在 传统 的 办 公 室 中 采用 各 种 
新 技术 、 新 机 器 、 新 设备 从 事 办 公 上 业务 ， 都 属于 办 公 自 动 化 的 领域 。 

办 公 自 动 化 系统 与 办 公 自 动 化 在 概念 上 存在 一 定 的 差别 。 办公 自 动 化 通常 指 办 
公 室 中 配备 具有 自动 化 功能 的 设备 , 这 些 设备 能 使 某 些 办 公 活 动 自动 化 或 实现 某 个 
单位 业务 的 自动 化 处 理 ; 而 办 公 自 动 化 系统 则 是 在 办 公 室 自动 化 功能 的 基础 上 发 展 
起 来 ， 以 办 公 自 动 化 技术 为 主体 同人、 组织、 制度 、 环 境 等 相 结合 的 完整 的 系统 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


办 公 自 动 化 管理 系统 的 开发 流程 

进一步 学 习 如 何 做 项 目 需求 分 析 与 系统 设计 
页 面 布 局 中 的 框架 布局 

使 用 递归 图 数 做 多 级 下 拉 荣 单 

系统 日 志 的 实现 


各 理 吾 有 吾 理 
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3.1 开发 背景 


随 着 中 国 加 入 WTO 及 全 球 经 济 一 体 化 进程 的 加 快 ， 世 界 经 济 已 由 工业 化 经 济 逐 步 进入 网 络 信息 
化 时 代 。 在 信息 时 代 来 临 之 际 ， 各 企业 都 紧 跟 时 代 的 脚步 ， 转 变 着 企业 的 经 营 模式 、 管 理 模式 ， 从 传 
统 的 人 工 管 理 体制 ， 向 信息 自动 化 管理 体制 过 渡 。 网 络 自动 化 办 公 系统 就 是 在 这 样 的 大 背景 下 应 运 而 


生 的 。 


信息 时 代 的 到 来 让 人 们 尝 到 了 “信息 爆炸 ”的 滋味 ， 信 息 的 大 量 涌 入 让 企业 在 信息 处 理 方面 应 接 
不 上 暇 ， 传 统 的 办 公 模 式 对 信息 的 处 理 方法 ， 早 已 不 能 满足 企业 对 信息 快速 、 准 确 的 处 理 的 要 求 。 对 信 
息 数 据 的 掌握 程度 、 处 理 能 力 ， 体 现 了 一 个 企业 对 市 场 的 敏感 程度 ， 数 据 的 真实 性 、 准 确 性 直接 决定 
着 企业 的 发 展 方向 。 从 传统 的 办 公 模 式 向 自动 化 办 公 管 理 模式 转变 ， 提 高 企业 的 信息 处 理 能 力 ， 以 增 
强 企业 的 市 场 竞争 力 ， 成 了 企业 发 展 过 程 中 的 首要 问题 。 


3.2 需求 分 析 


根据 与 客户 的 多 次 交谈 和 了 解 ， 本 系统 所 面向 的 客户 对 象 的 情况 如 下 : 


办 办 办 办 罗 


加 


加 


3.3.1 


所 属 IT 行业 ， 目 前 主要 以 开发 门户 网 、 企 业 网 等 中 小 型 网 站 为 主 。 

公司 经 过 多 年 经 营 ， 已 经 相对 稳定 ， 并 有 了 自己 的 网 站 、 企 标 、 规 章 制度 和 基本 架构 。 
公司 规模 30 一 100 人 ， 人 和 手 一 机 ， 主 要 包括 部 门 经 理 、 普 通 职员 、 试 用 人 员 和 实习 人 员 。 
职员 所 用 计算 机 属于 局 域 网 内 网 ， 不 允许 访问 外 网 。 

公司 暂时 分 为 技术 部 、 人 事 部 和 质量 部 ， 但 不 排除 后 期 增加 其 他 部 门 的 可 能 。 

公司 实行 人 性 化 管理 ， 允 许 员 工 自 由 发 表意 见 和 想法 ， 还 有 丰富 多 彩 的 活动 ， 如 比赛 、 旅 
游 等 。 

为 了 提高 工作 效率 ， 所 有 职员 定期 写 工作 计划 ， 包 括 周 计划 、 月 计划 ， 部 门 经 理 还 有 年 计划 
和 任务 计划 。 

根据 计划 的 完成 程度 和 完成 质量 ， 不 定期 地 选取 优秀 员工 。 


3.3 系统 设计 


系统 目标 


本 系统 是 针对 中 小 型 企业 内 部 自动 化 办 公 管 理 的 要 求 而 设计 的 ， 主 要 实现 如 下 目标 ; 


键盘 操作 ， 人 快速 响应 。 
实现 文件 类 信息 的 强大 的 管理 能 力 。 
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实现 对 员工 基础 信息 ‘人事 消息 ) 的 管理 功能 等 。 
实现 个 人 办 公 的 信息 自动 化 管理 功能 。 

发 布 会 议 信息 ， 并 对 会 议 信息 进行 管理 。 

对 系统 用 户 进行 管理 。 

为 了 加 强 数据 保密 性 ， 为 每 个 用 户 组 设置 权限 级 别 。 
系统 最 大 限度 地 实现 了 易 安装 性 、 易 维护 性 和 易 操作 性 。 
系统 运行 稳定 、 安 全 可 靠 。 


办 办 办 办 办 欠 风 


3.3.2 ”系统 功能 结构 


根据 系统 分 析 ， 下 面 给 出 系统 的 前 、 后 台 功 能 结构 图 。 
办 公 自 动 化 前 台 功 能 结构 图 如 图 3.1 所 示 。 


图 3.1 办 公 自 动 化 前 台 管理 系统 
办 公 自 动 化 后 台 功能 结构 图 如 图 3.2 所 示 。 
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一 | 系统 日 志 
一 数据 备份 
| 修改 密码 


图 3.2 办 公 自动 化 后 台 管理 系统 
3.3.3 ”系统 功能 预览 


办 公 自 动 化 管理 系统 由 多 个 功能 模块 组 成 ， 下 面 仅 列 出 几 个 典型 功能 的 页 面 ， 其 他 页 面 可 参见 光 
盘 中 的 源 程序 。 

前 台 登 录 界面 如 图 3.3 所 示 , 该 页 面 用 于 实现 对 用 户 登录 的 用 户 名 和 密码 进行 验证 。 企业 信息 页 面 
如 图 3.4 所 示 ， 该 页 面 用 于 显示 企业 文化 和 各 规章 制度 。 


图 3.3 前 台 登 录 (光盘 \TMW3\oavindex.php) 3.4 ”规章 制度 (光盘 \TM\03\oa\qyxx\r_system.php) 
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个 人 计划 页 面 如 图 3.5 所 示 , 该 页 面 主 要 实现 用 户 的 工作 反馈 和 周 、 月 计划 等 功能 。 后 台 登 录 界面 
如 图 3.6 所 示 ， 该 页 面 用 于 管理 员 登 录 后 台 。 


et arm MHI 全 


arm 。。 sadfrwtrwlt 如 和 主 
-OTE mana … 理 玉 


dd 
上 和 于 计划 El 


图 3.5 个 人 计划 (光盘 \TM\03\oa\grjh\person_plan.php) 图 3.6 后 台 登 录 (光盘 \TMW3\oavadmin\index.php) 


职员 管理 页 面 如 图 3.7 所 示 ， 该 页 面 主要 用 于 对 职员 的 查询 与 修改 等 功能 。 权 限 分 配 页 面 如 图 3.8 
所 示 ， 该 页 面 用 于 对 功能 使 用 权 的 分 配 。 


图 3.7 职员 管理 (光盘 \TM\WO3\oavadminvzyglishow_stafphp) 图 3.8 权限 分 配 (光盘 \TMW3\oavadmin\qxgl\pur_assign.php) 


3.3.4 ”系统 流程 图 


用 户 在 登录 一 个 系统 后 ， 会 进行 一 系列 的 操作 ， 把 这 些 操作 的 过 程 和 结果 以 图 形 的 形式 表现 出 来 ， 
这 就 是 系统 流程 图 。 一 个 好 的 流程 图 ， 不 仅 可 以 让 开发 者 迅速 地 理 清 思路 ， 及 时 解决 出 现 的 问题 ， 也 
可 以 让 使 用 者 很 快 明白 该 系统 的 操作 方式 与 方法 。 下 面 给 出 办 公 自 动 化 管理 系统 的 工作 流程 图 ， 如 
图 3.9 所 示 。 


二 
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3.3.5 


3.9 办公 自动 化 管理 系统 流程 图 


开发 环境 


在 开发 办 公 自 动 化 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 


二 


办 办 办 办 办 凶 


。 服务器 端 


操作 系统 : Windows 2003 Server。 

服务 器 : Apache 2.0。 

PHP 软件 : PHP 5.0。 

数据 库 : MySQL 5.0。 

MySQL 图 形 化 管理 软件 : PhpMyAdmin-2.5.5。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0。 

分 辨 率 : 最 佳 效果 1024X768 像素 。 


141 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


2 客户 端 
回 ”浏览 器 : 推荐 使 用 下 6.0 及 以 上 版 本 。 
回 ”分辨 率 : 最 佳 效果 1024X768 像素 。 


3.3.6 ”文件 夹 组 织 结构 
系统 功能 结构 和 系统 流程 设计 完成 后 ， 接 下 来 就 要 设计 网 站 的 文件 夹 结构 了 ， 合 理 的 文件 夹 结构 


不 仅 易 于 快速 地 开发 ， 对 后 期 系统 的 调试 、 维 护 和 管理 也 能 起 到 事半功倍 的 效果 。 本 系统 的 文件 夹 组 
织 结构 如 图 3.10 所 示 。 


量 

部 
OR 
四 
济 


日 -和 o 办 公 目 动 化 系统 根 目录 
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图 -图 -图 -图 -图 -图 苹 - 图 -图 -图 -外 
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前 台数 据 库 链接 文件 
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3.10 办公 自动 化 管理 系统 文件 夹 组 织 结构 
3.4 数据 库 设 计 
基于 目前 的 系统 开发 ， 没 有 数据 库 的 支持 根本 是 无 法 想象 的 ， 办 公 自 动 化 管理 系统 更 是 使 用 了 大 
量 的 数据 表 来 存储 数据 。 本 节 就 对 办 公 自 动 化 管理 系统 所 需要 使 用 的 数据 库 进 行 说 明 与 设计 。 
3.4.1 数据 库 分 析 
由 于 本 系统 采用 的 是 PHP 语言 ， 数 据 库 理所当然 地 使 用 MySQL， 不 仅 是 因为 开发 成 本 低 ， 更 重 


要 的 是 两 者 之 间 的 默契 程度 和 稳定 程度 要 远 远 高 于 其 他 的 数据 库 组 合 。 对 于 一 个 企业 的 内 部 办 公 自 动 
化 管理 系统 是 完全 够 用 了 。 
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3.4.2 ”数据 库 概念 设计 


根据 以 上 的 需求 分 析 及 系统 分 析 ， 规 划 出 办 公 自 动 化 管理 系统 主要 的 几 个 实体 关系 E-R 图 。 
1. 用户 信息 实体 
用 户 信息 实体 包括 职员 的 账号 、 密 码 、 姓 名 、 性 别 等 表示 个 人 身份 的 数据 资料 ， 如 图 3.11 所 示 。 


图 3.11 用 户 信息 实体 E-R 图 
2. 考勤 登记 实体 
考勤 登记 实体 包括 登记 日 期 、 登 记 时 间 、 登 记 类 型 (上班 或 下 班 》、 登 记 状 态 (迟到 或 早退 ) 和 


登记 人 等 信息 ， 如 图 3.12 所 示 。 


登记 人 


3.12 考勤 登记 实体 ER 
3.4.3 数据库 物理 结构 设计 


在 本 系统 中 创建 了 一 个 数据 库 db_office， 一 共 包含 13 个 数据 表 ， 如 图 3.13 所 示 。 
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图 服务 器 : localhost ， 数据 库 : db_ofice 


表 类 型 
th_company = MylSAM 
th_controller = MylSAM 


th_depart MySAM 
tb_group MISAM 
tb iss MISAM 
th_list MMSAM 
th_yb MISAM 
th_person MySAM 
th_plan MyISAM 
th_register MyISAM 
th_setup MySAM 
th_superson = MyISAM 
tb users MyISAM 


整理 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
9b2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_ci 
9b2312_chinese_ci 
gb2312_chinese_ci 


说 明 
企业 信息 列表 
管理 员 列表 
部 门 管理 列表 


图 3.13 db_office 库 中 的 数据 表 列 表 
限于 篇 幅 ， 这 里 只 给 出 主要 的 表 结 构 和 表 说 明 ， 其 他 数据 表 结构 请 参见 本 书 附带 的 光盘 。 
1. tb_users (用户 列表 ) 


用 户 列 表 主 要 用 于 存储 职员 的 姓名 、 性 别 等 私人 信息 及 部 门 、 工 作 组 等 与 公司 相关 的 公共 信息 。 
该 数据 表 结 构 如 图 3.14 所 示 。 
图 服务 器 : localhost > 怕 数据 库 : db_office ， 国 表 :tb_users 


字段 类 型 整理 尾 性 Null 默认 额外 说 明 
过 int(4) 否 auto_increment 自动 编号 
UL_user varchar(50) gb2312_chinese_cl 理 用 户 账号 
upwd varchar(20) gb2312_chinese_ci 否 用 户 密码 
uname varchar(20) gb2312_chinese_cl 是 NULL 用 户 姓 名 
UL_sex charll) gb2312_chinese_cl 是 NULL 用 户 性 别 
ubirth date 是 NULL 用 户 生日 
uaddress varchar(50) gb2312_chinese_cl 是 WULL 用 户 住址 
utel varchar(20) gb2312_chinese_cl 是 NULL 用 户 电话 
uemail varchar(50) gb2312_chinese_cl 是 NULL 用 户 Enail 
udepart varchar(20) gb2312_chinese_ci 理 所 属 部 门 
is_on int(1) 否 测 活 账号 

图 3.14 用 户 列表 


2. tb_person (人 事 列表 ) 


人 事 列 表 用 于 存储 人 事 部 门 发 布 的 信息 ， 如 信息 标题 、 信 息 内 容 、 信 息 时 间 等 。 该 表 结构 如 图 3.15 
所 示 。 


园 服务 器 : localhost 》 章 数据 库 : db_office ， 国 表 :tb_person 


字 及 类 型 整理 性 性 Null 默认 铬 外 说 明 
刘 int(4) 否 auto_increment 。 自动 编号 
pite varchar(50) gb2312_chinese_ci 理 公告 标 是 
p_content mediumtext gb2312_chinese_ci 理 公告 内 容 
ptime date 理 发 布 时 间 
uid intg) 再 消息 类 别 

图 3.15 人 事 列表 
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3.5 ”公共 模块 设计 


本 系统 的 公共 模块 包含 conn 数据 库 链接 文件 、css 样式 文件 、js 脚本 文件 和 inc 下 的 自 定义 函数 文 
件 和 包含 文件 等 几 类 文件 , 其 中 数据 库 连 接 文件 和 ess 样式 文件 在 前 面 的 章节 中 有 过 系统 的 介绍 了 , 这 
里 主要 讲解 本 系统 所 涉及 到 的 js 脚本 文件 和 部 分 自 定义 函数 文件 。 


3.5.1 JavaScript 脚本 


在 办 公 自 动 化 管理 系统 中 ，JavaScript 脚本 一 般 用 于 表单 元 素 验证 ， 如 判断 text 文本 框 输入 是 否 为 
空 ， 输 入 格式 是 否 符合 标准 等 。 在 网 页 中 使 用 JavaScript 脚本 的 方式 主要 有 3 种 : 
(1) 在 网 页 中 使 用 <script></scrip 人 标签 对 
<script></scrip 人 > 标签 对 可 以 放 在 网 页 的 任意 位 置 ， 一 般 是 放 在 <head></head> 或 <body></body> 之 
间 ， 代 码 如 下 : 
<head> 
<script>document.write(" 办 公 自 动 化 管理 系统 开发 "):</script> 
</head> 
(2) 在 单独 文件 中 使 用 
如 果 JavaScript 脚本 比较 多 ， 而 且 位 置 分 散 不 易 管理 ， 可 以 统一 放 到 一 个 扩展 名 为 js 的 文件 中 ， 
使 该 文件 成 为 JavaScript 脚本 文件 。 在 脚本 文件 中 ,不 需要 使 用 <script></script> 标 签 对 ， 直 接 写 脚本 代 
码 即 可 。 当 有 页 面 需要 使 用 到 里 面 的 JavaScript 脚本 时 ， 可 以 这 样 引用 ， 代 码 如 下 : 
<script src="ad js.js" jangauge="javascript"></script> 
(3) 在 表单 元 素 或 标签 中 使 用 
这 是 最 直接 的 使 用 方式 ， 如 果 是 少量 的 脚本 则 可 以 这 样 使 用 。 例 如， 在 超 链接 标签 <a> 中 想 要 使 用 
JavaScript 脚本 ， 代 码 格式 如 下 : 
<a href="#" onclick="alert(hello):">hello</a> 
在 办 公 自 动 化 管理 系统 中 ,这 3 种 方法 都 可 以 使 用 到 ， 在 后 面 涉 及 到 具体 应 用 的 时 候 再 进行 说 明 。 
下 面 先 来 看 两 个 经 常 使 用 到 的 脚本 函数 。 
(1) 验证 函数 ， 用 于 判断 表单 元 素 是 否 为 空 。 如 果 为 空 ， 则 返回 false， 并 将 光标 焦点 定位 到 出 问 
题 的 表单 元 素 。 程 序 代码 如 下 : 
例 程 01 代码 位 置 : 光盘 \TM\03\oa\js\client jsjs 


// 后 台 登 录 界 面 验 证 脚本 
fnction checkO{ 
if(login.username.value—""){ // 判 断 用 户 名 是 否 为 空 
alert(" 请 输入 用 户 名 !!1"); 
login.username .focus(): // 将 光标 焦点 定位 到 该 表单 元 素 
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Teturn false: 

b 

iflogin pwd.value—""){ // 判 断 用 户 密码 是 否 为 空 
alert(" 请 输入 密码 !1"); // 如 果 为 空 ， 弹 出 提示 框 
login pwd .focusO:; // 将 光标 焦点 定位 到 该 表单 元 素 
Teturn false: // 返 回 false 


a 
| 


(2) 删除 确认 ， 当 要 执行 删除 操作 时 ， 使 用 JavaScript 脚本 文件 进行 确认 ， 以 免 因为 误 操作 而 引 
起 不 必要 的 损失 ， 代 码 如 下 : 


例 程 02 ”代码 位 置 : 光盘 \TM\03\oa\js\client js.js 


function cftmO{ 
这 confimm( 确 认 要 删除 吗 ? ")) /选择 框 函数 
Teturm true: /如 果 选 择 确认 ， 则 返回 tue， 继 续 执行 
else 
return false: // 否 则 ， 返 回 false 


} 
3.5.2” 自 定 义 函 数 


在 inc 文件 夹 下 ， 有 两 个 主要 文件 : chec.php 文件 和 func.php 文件 ， 其 中 chec.php 文件 是 权限 检查 
文件 ， 是 办 公 自 动 化 管理 系统 每 个 页 面 都 要 引用 的 ，func.php 文件 是 自 定义 函数 文件 。 


1. chec.php 文件 


对 办 公 自动 化 管理 系统 来 说 ， 用 户 对 每 个 页 面 的 访问 都 要 有 该 页 的 权限 才 可 以 ， 如 果 没 有 权限 级 
别 的 限制 ， 随 意 访问 重要 的 资源 和 数据 ， 那 么 只 能 说 明 这 是 个 完全 失败 的 系统 。chec.php 文件 的 代码 
如 下 : 


例 程 03 ”代码 位 置 : 光盘 \TM\03\oa\inc\chec.php 
<?php 
session start(): // 开 启 session 支 持 
© ifllisset($ SESSION[u name])) 
echo "<script>alert( 您 无 权 访问 "):location="../index.php':</script>"; 
© ifs_ SERVER['HTIP REFERER']—"") 
echo "<script>alert(' 本 系统 不 允许 从 地 址 栏 访问 "):history.go(-1):</script>"; 
?> 


< 代码 贴 二 

@ isset0 函 数 : 用 来 判断 session[u name] 是 否 被 创建 ， 如 果 session 已 经 创建 ， 返 回 true， 否 者 返回 false。 其 中 
session[u_ name] 是 用 户 登录 验证 通过 后 创建 的 ， 如 果 没 有 登录 的 步骤 或 session 超时 ， 那 么 就 会 提醒 用 户 没有 权限 。 

@ $_ SERVER[HTTP REFERER]: 系统 预定 义 变量 ， 存 储 的 是 上 一 页 的 URL 地 址 。 这 个 判断 的 作用 是 防止 当前 用 
户 访问 不 属于 自己 权限 内 的 网 页 。 

2. func.php 文件 

限于 篇 幅 ， 这 里 只 给 出 两 个 自 定义 函数 的 代码 和 说 明 ， 其 他 函数 请 在 光盘 中 自行 查看 。 
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读 取 字 段 函 数 read_field0 
本 函数 的 作用 是 根据 传 入 的 id 值 取得 相关 的 字段 名 ， 一 共有 4 个 参数 ， 函 数 的 代码 如 下 : 


例 程 04 ”代码 位 置 : 光盘 \TMWO3\oavincfinc php 


function read _ field($conn.Stablename.$fieldname,Sn id){ 
S$sqlstr = "select ".$fieldname." from ".$tablename." where id=".$n id: ” // 生 成 sql 语 句 


Sresult = mysql_ query($sqlstr$conn): /执行 sql 语 句 
Srows = mysql_fetch row(Sresult): // 返 回 结果 集 
Tetum Srows[0]: // 返 回 需要 的 字段 名 


read field0) 函 数 的 参数 说 明 如 表 3.1 所 示 。 
表 3.1 read_field() 函 数 参数 说 明 


参数 说 明 
数据 库 链接 资源 变量 


Sfeldname 要 查找 的 字段 名 


要 查找 的 id 值 


处 理 消息 函数 re_message() 
本 函数 的 作用 是 返回 消息 结果 ， 一 共有 2 个 参数 ， 函 数 的 代码 和 参数 解释 如 下 : 


例 程 05 ”代码 位 置 ; 光盘 \TM\03\oa\admin\inc\func.php 
function re_ message($result,$] address){ 


if($result) 
echo "<script>alert( 操 作成 功 ! "):location=".$l_address.";</script>"; /如 果 结 果 为 tue， 转 到 其 他 页 面 


else 


echo "<script>alert( 系 统 繁忙 ， 请 稍 后 再 试 '):history.go(-1):</script>"; 。“ // 如 果 结 果 为 false， 回 到 上 一 步 


> $result: 数据 库 返 回 结果 ， 如 果 返 回 值 为 tue， 说 明 对 数据 库 的 操作 成 功 ， 如 果 返 回 值 为 
false， 说 明 操 作 失 败 。 
> 。$l address: 操作 成 功 后 返回 的 url 地 址 。 


3.6 前 台 首 页 设计 


对 于 一 个 办 公 自 动 化 管理 系统 来 说 ， 首 页 不 可 能 ， 也 不 允许 显示 太 多 的 内 容 和 复杂 的 设计 ， 因 为 
办 公 自 动 化 管理 系统 的 作用 是 帮助 提高 企业 的 管理 效率 和 职员 的 工作 效率 。 如 果 页 面 过 于 繁琐 ， 会 让 
职员 分 不 清 主 次 ， 找 不 到 重点 。 所 以 一 定 要 突出 重点 内 容 ， 显 示 关 键 功 能 。 
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根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 , 本 系统 的 首页 面 中 要 显示 的 模块 主要 有 3 部 分 : 
回 ”网 站 首页 导航 栏 : 包括 当前 用 户 、 所 在 部 门 、 当 前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 
回 ”网 站 左 侧 导 航 栏 : 包括 各 个 管理 模块 及 分 类 。 
> ”企业 信息 模块 : 包括 公司 简介 、 规 章 制度 、 组 织 结构 和 企业 管理 。 
企业 绩效 模块 : 包括 任务 绩效 、 质 量 绩效 和 绩效 评定 。 
人 事 消 息 模 块 : 包括 企业 公告 、 活 动 安排 和 消息 管理 。 
审核 批示 模块 : 包括 发 布 审核 和 批示 审核 。 
考勤 管理 模块 : 包括 上 下 班 登记 、 病 事假 登记 、 加 班 登记 和 考勤 设置 。 
个 人 计划 模块 : 包括 工作 反馈 、 周 计划 、 月 计划 、 年 计划 和 任务 计划 。 
职员 天 地 模块 : 包括 职员 浏览 、 意 见 箱 和 个 人 设 定 。 
回 ”网 站 主 显示 区 : 默认 显示 企业 公告 、 活 动 安排 和 个 人 计划 及 批示 审核 。 
下 面 看 一 下 本 案例 中 提供 的 前 台 首 页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\03\oa\pub_main.php， 
如 图 3.16 所 示 。 


VY YY ™Y 


v 


办 公 自动 化 管理 系统 O 人 A 


| zi 明天 下 午 3 点 ,羽毛球 第 三 过 抽签 


2007-12-04 副 斌 


| 
主 显示 区 


- 审核 批示 
类 型 “日 期 标题 是 否 所 示 。 操作 


左 侧 导 航 栏 2007-12-17 本 月 5 日 夫 至 ， 完 成 影 .查看 全 文 任务 计划 | 2007-12-17 一 个 新 方向 。 未 审核 “修改 | | 风 除 
2007-12-17 本 周 这 成 影视 在 线 项 目 .查看 全 文 周 计 划 2007-12-17 统 等 规划 节约 资 未 审核 修改 | | 内 


图 3.16 办 公 自动 化 管理 系统 前 台 首 页 
3.6.2 前台 首页 技术 分 析 
目前 网 页 页 面 的 布局 主要 有 两 种 方式 : 表格 布局 与 框架 布局 。 本 系统 前 台 页 面 的 布局 使 用 的 是 框 


架 布 局 。 框 架 布局 就 是 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 ， 
网 页 之 间 是 相互 独立 的 ， 没 有 直接 的 关联 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 
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示 在 浏览 者 的 浏览 器 中 。 框 架 布局 的 好 处 是 : 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框 
架 页 面 ， 其 他 子 页 面 保持 不 变 。 使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 

下 面 来 具体 看 一 下 框架 布局 的 使 用 格式 及 属性 。 

1. 框架 布局 格式 

框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 ， 常 用 的 格式 如 下 : 

<html> 

<head> 


</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 


<body> 


</body> 
</noframes> 
</html> 


其 中 <frameset> 和 <frame> 标 签 是 框架 集 标 记 ， 而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 
实行 的 一 种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 记 里 的 内 容 ， 让 用 户 能 够 正 
常 浏览 网 页 。 

2. 框架 集 属性 

框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
FRAMESET 标记 的 属性 对 框架 的 结构 进行 设置 。 下 面 给 出 框架 集 的 常用 属性 值 、 说 明和 应 用 举例 ， 如 
表 3.2 所 示 。 


表 3.2 框架 集 的 常用 属性 


属性 名 称 属性 说 明 应 用 举例 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 取 | <frameset cols="25%.100.*" > 
COLS 值 有 3 种 形式 : 像素 、 百 分 比 (%) 和 相对 | <frame></frame> 


尺寸 5 </frameset> 


<frameset rows="25%,100.*" > 


在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 取 | <frame> 


ROWS 


值 和 COLS 类 似 ， 也 是 3 种 形式 <frame> 
</frameset> 
<framset cols="2590.*" cols="*" frameborder="0"> 
定 周围 是 示 , ( 
ODE 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 ( 显 | .. 


示 边 框 ， 默 认 值 ) 或 0〈 不 显示 边框 7 


</frameset> 
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续 表 


属性 名 称 属性 说 明 应 用 举例 
<framset cols="25%,*" cols="*" 
指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 默 认 | framespacing="1"> 

是 无 间隔 的 5 


FRAMESPACING 


</frameset> 

<framset cols="25%,*" cols="*" frameborder="1" 
指定 边框 的 宽度 ，frameborder 属性 为 1 时 该 | border="5"> 

属性 才 有 效 8 


BORDER 


</frameset> 


3. 框架 属性 
使 用 FRAME 标记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 框 架 是 否 包含 滚动 条 以 及 在 框架 中 显 
示 的 网 页 等 。FRAME 标记 的 常用 属性 及 其 说 明 如 表 3.3 所 示 。 


表 3.3 框架 的 常用 属性 


属性 名 称 属性 说 明 

NAME 指定 框架 的 名 称 

SRC 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、PHP、JSP 等 网 页 文件 ) 
FRAMEBODER 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 为 默认 ) 或 0 (不 显示 边框 
NORESIZE 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 

SCROLLING 指定 框架 是 否 包含 滚动 条 。 属 性 可 以 是 yes (有 ) 、no (没有 ) 和 auto( 自 由) 


3.6.3 前台 首 页 的 实现 过 程 


国 前 台 首页 使 用 的 数据 表 : tb_users、tb list、tb_person、tb register、tb_plan 

办 公 自 动 化 管理 系统 的 前 台 首 页 采用 的 二 分 栏 结构 布局 ， 二 分 栏 布局 的 特点 是 简洁 、 大 气 、 个 性 
鲜明 ， 其 框架 设计 依据 其 内 容 形式 的 变化 而 灵活 多 变 ， 结 构 简 练 。 非 常 符合 办 公 自 动 化 管理 系统 的 风 
格 ， 本 系统 中 框架 布局 的 具体 应 用 如 下 。 

例 程 06 ”代码 位 置 : 光盘 \TM\W3\oapub_ main php 


© <framesetrows="111,*" cols="*" frameborder="no" border="0" framespacing="0"> 
© <frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
目 <frameset rows="*" cols="220,*" framespacine="0" frameborder="not" border="0"> 
9 <frame src="left.php" name="leftFrame" scrolling="auto" noresize="noresize" id="leftFrame" title="leftFrame" /> 
© <frame src="main.php" name="mainFrame" scrolling="auto" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
@ <noframes><body> 
</body> 
</noframes> 
< 代码 贴 十 


@ Iows="111.*" cols="*": 定义 顶部 框架 。 
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@ src="top.php": 定义 顶部 框架 内 显示 的 页 面 。 

图 rows="*" cols="220.*": 定义 下 部 左 侧 框架 。 

@ src="left.php": 定义 左 侧 框架 显示 的 页 面 。 

@ src="main.php": 定义 中 间 主 框架 显示 的 页 面 。 
@ <noframes>: 当 浏 览 器 不 支持 框架 集 时 的 内 容 。 


3.7 人 事 消息 模块 设计 


人 事 消息 模块 主要 是 对 文件 进行 收发 管理 ， 模 块 的 设计 和 实现 并 不 是 十 分 复杂 ， 但 在 整个 系统 中 
的 位 置 却 很 重要 。 因 为 公告 栏 和 意见 箱 面向 的 是 全 体 用 户 ， 企 业 的 最 新 动向 和 职员 的 意见 想法 首先 都 
是 在 这 里 体现 出 来 的 。 
3.7.1 人 事 消 息 模块 概述 


人 事 消息 模块 主要 包含 了 两 部 分 内 容 ， 一 部 分 是 面向 全 体 用 户 的， 包括 查看 公告 、 活 动 、 意 见 箱 
等 ， 一 部 分 是 仅 对 人 事 部 开放 ， 其 他 用 户 不 允许 、 也 不 会 看 到 的 页 面 ， 如 公告 管理 、 意 见 管理 等 。 下 


面 给 出 人 事 消息 框架 ， 如 图 3.17 所 示 。 


回复 留言 ) 4 一 一 一 Cs 


图 3.17 ”人事 消 息 用 例 图 
3.7.2 人事 消息 模块 技术 分 析 


在 人 事 消息 模块 中 ， 企 业 公 告 和 活动 安排 都 是 在 同一 显示 页 中 显示 (p_message.php)。 为 了 


x 
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不 同 的 操作 ， 需 要 给 显示 页 (p_message.php) 传 一 个 u id 值 ， 在 数据 表 中 根据 不 同 的 u id 值 ， 取 得 不 
同类 别 的 内 容 。 传 值 方式 有 url 传 值 方 式 ( 如 p_message.php?u id=1 ) 和 表单 传 值 方式 两 种 。 

同样 的 ， 在 接收 页 中 ， 接 收 方式 也 有 两 种 : get 方式 和 post 方式 。PHP 中 使 用 预定 义 变量 $_GET 
和 $_POST 来 接收 传 值 ， 格 式 为 : 


$_GET/POST[u id]; /注意 ， 变 量 名 都 为 大 写 


对 于 ml 的 传 值 方式 , 接收 页 始终 用 $_GET[] 变 量 来 接收 , 如 果 使 用 form 表单 传 值 ， 那 么 就 看 form 
表单 中 method 属 性 的 设置 了 。 如 果 method=get, 就 使 用 $_GET[] 变 量 ;如果 method=post, 就 使 用 $_POST[] 
变量 。 

PHP 对 传 值 方 式 和 接收 方式 规定 很 严格 ， 用 $_POST[u_id] 接 收 不 了 p_message.php?u_id=1 传 过 来 
的 参数 。 同 样 ， 用 $_GET[u_id] 也 接收 不 着 method=post 的 表单 元 素 值 。 

名 注意: 如 果 在 php.ini 配置 文件 中 GLOBAL=ON/OFF 一 行 的 值 为 ON， 那 么 直接 写 $name 就 可 以 
调用 表单 元 素 的 值 ， 而 不 区 分 get 和 post; 如 果 为 OFF， 则 不 可 以 。 直 接应 用 表单 名 称 十 分 
方便 ， 但 也 存在 着 安全 隐患 。 推 荐 关闭 GLOBAL。 


3.7.3 ”消息 管理 的 实现 过 程 


国 ”消息 管理 模块 使 用 的 数据 表 : tb_person 

选择 “人 事 消 息 ” 下 拉 列 表 的 “消息 管理 ”选项 ， 在 主 框架 (mainFrame) 内 就 会 显示 消息 管理 页 
面 ， 在 页 面 中 显示 了 发 布 过 的 消息 列表 和 对 消息 的 操作 ， 包 括 “ 修 改 ”、“ 删 除 ” 和 “发 布 新 消息 ”。 
消息 管理 页 面 的 运行 结果 如 图 3.18 所 示 。 


图 3.18 消息 管理 页 面 运行 结果 


在 本 页 面 中 , 系统 首先 通过 权限 检查 文件 (chec.php) 判断 用 户 是 否 为 登录 用 户 , 是否 为 非法 链接 。 
如 果 被 系统 判断 为 非法 操作 ， 系 统 将 直接 关闭 ; 如 果 没 有 异常 情况 ， 系 统 才 继 续 执行 下 面 的 代码 (在 
后 面 的 例 程 中 ， 几 乎 每 个 页 面 都 有 chec.php 文件 ， 如 果 不 是 特殊 说 明 ， 则 都 为 正常 登录 ， 到 时 将 不 再 
具体 说 明 ) ， 当 用 户 通过 检查 后 ， 系 统 将 访问 数据 库 ， 从 人 事 消息 数据 表 中 〈tb_person) 读 取出 消息 
发 布 时 间 、 消 息 标题 ， 显 示 在 页 面 中 ， 并 通过 id 为 每 条 数据 设置 “修改 ”和 “删除 ”操作 。 程 序 关键 
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代码 如 下 : 
例 程 07 代码 位 置 光盘 \TM\03\oa\rsxx\p_manage.php 
<?php 
session start(): // 开 启 session 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php": // 包 含 数据 库 连 接 文 件 
?> 
<link href="../css/style.css" rel="stylesheet" /> /引入 外 部 CSS 样 式 文件 
<script src="../ijs/client js.js"></script> /引入 js 文件 
<?php 
/* 显 示 全 部 消息 记录 六 
S$sqlstr = "select id.p_time.p_title from tb_person": // 生 成 sql 语 句 @ 
Sresult = mysql_query($sqlstr, $conn): // 执 行 sql 语 句 ， 生 成 结果 集 
while($rows = mysql_fetch row(Sresulb){ // 循 环 输出 结果 集 记 录 


echo "<tr>"; 
for($i=1:$i<count($rows):$i++){ 
echo "<td height=30 style='text-indent: 30px:'>".Srows[Si]."</td>"; 


} 
echo "<td><a href="m_message.php?id=".Srows[0]."'> 修 改 </a>/<a 
href="'d_message_chk.php?id=".$Srows[0]."" onclick="return del_mess0;> 删 除 </a></td>"; 
echo "</tr>";} 
?> 
<h> 
<td height="30" align="right" valign="middle" colspan="3"><a href='add_manage.php' target="mainFrame"> 发 
布 新 消息 </a></td> 


</> 


下 面 分 别 介绍 “添加 新 消息 ”、“ 修 改 ” 和 “删除 ”消息 的 实现 过 程 。 
1. 发 布 新 消息 
单 击 “ 发 布 新 消息 ” 超 链接 ， 进 入 发 布 页 面 。 发 布 消息 页 面 的 运行 结果 如 图 3.19 所 示 。 


wea OA 


EE 


区 
图 3.19 发布 消息 页 面 运行 结果 
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发 布 页 面 中 ， 主 要 包含 一 个 消息 表单 元 素 ， 部 分 表单 元 素 如 表 3.4 所 示 。 
表 3.4 发 布 消息 页 面 的 主要 表单 元 素 


addmess action="add manage chk php" method="post" id="addmess" 发 布 消息 表单 
_Putitle text | name="p title" id="p title” | 消息 标题 
_P_content textarea | name="p_content" id="p_content" cols="60" rows="15" | 消息 内 容 
<option value="9"> 企 业 公 告 </option> a 
p_type select ds 消息 类 型 


<option value="10"> 活 动 安排 </option> 
Value=" 发 布 " onclick="return add mess0:" 


Submit2 


submit 


“发 布 ”按钮 


在 消息 页 面 中 输入 消息 的 相关 内 容 后 ， 单 击 “ 发 布 ” 按钮 ， 系 统 将 会 跳 到 消息 处 理 页 (add_manage 
_chkphp) 进行 处 理 ， 将 消息 存储 到 数据 库 中 ， 然 后 再 返回 该 页 。 程 序 关键 代码 如 下 : 


例 程 08 ”代码 位 置 : 光盘 \TM\03\oa\rsxx\add_manage_chk.php 


<2php 
session start(); // 开 启 session 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php": /包含 数据 库 链 接 文件 
include "../inc/func.php"; // 包 含 函数 处 理 文件 
”添加 新 消息 */ 


Sresult = mysql_query($sqlstr,$conn):; 
re_message(Sresult,"p_manage.php"); // 消 息 处 理 函数 


/这 守 突 详实 兴 兴 光 突 详实 次 / 


?> 
2. 修改 消息 


消息 发 布 后 ， 可 以 随时 对 消息 内 容 进行 修改 。 在 消息 管理 页 面 中 ， 选 择 要 修改 的 消息 列 ， 单 击 修 
改 超 链接 将 进入 消息 修改 页 面 。 在 消息 修改 页 面 中 ， 有 一 个 和 发 布 页 面 非常 相似 的 “修改 ”表单 。 消 
息 修改 页 面 中 的 主要 表单 元 素 如 表 3.5 所 示 。 


表 3.5 修改 页 面 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
addmess form action="m _message_chk.php" method="post" 修改 消息 表单 
_Pptitle text value="<?php echo $rows[1]: ?>" 消息 标题 


_P_content textarea <?php echo Srows[2]: ?> 消息 内 容 
<?phpif($rows[4] 一 "9"){?> 


<option value="9" selected="selected"> 企 业 公 告 </option> 

<option value="10"> 活 动 安排 </option> 

p_type <?php}else{?> 消息 类 型 
<option value="9"> 企 业 公告 </option> 

<option value="10" selected="selected"> 活 动 安排 </option> 
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续 表 
名 称 | 元 素 类 型 重要 属性 含义 
id hidden | value="<?php echo Srows[0]: ?>" 修改 消息 id 
submit2 submit Value=" 修 改 " onclick="return add_mess():" “修改 ”按钮 


消息 修改 完成 后 ， 单 击 “ 修 改 ” 按 钮 ， 在 m_message_chk.php 页 中 将 会 处 理 修改 事件 ， 通 过 隐藏 
表单 传 过 来 的 id 定位 到 要 修改 的 消息 项 ， 使 用 update 语句 实现 对 数据 的 更 新 。 程 序 的 关键 代码 如 下 : 
例 程 09 ”代码 位 置 : 光盘 \TM\03\oa\rsxx\m_message_chk.php 


<?php 
session start(): /开启 session 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../conn/conn.php":; /包含 数 据 链接 文件 
/* 更 新 数据 库 */ 


$sqlstr = "update tb_person set p_title =".$ POSTIp title].",p_content =".$ POST[p_content].",p_time =nowO,u id = 
"$_POST[p_ typel" where id=".$ POST[id]: 
Sresult = mysql_query($sqlstr,$conn); 
re_message(Sresult,"p_manage.php"); // 消 息 处 理 函 数 


/这 守 突 内 风光 兴 兴 内 突 突 家 闪 内 / 


?> 
3. 删除 消息 


最 后 来 看 一 下 删除 操作 。 每 隔 一 段 时 间 管 理 员 就 要 定期 清理 一 下 无 用 的 消息 和 活动 安排 ， 以 加 速 
页 面 的 浏览 速度 ， 减 轻 服务 器 压力 。 删 除 消息 的 运行 结果 如 图 3.20 所 示 。 


2001-11-16 明天 下 和 3 点 ， 羽 所 于 第 三 轮 抽 答 jms 
| 2001-11-15 | 13 昌 考试 REYTE ET l ba 
2007-11-27 


图 3.20 ”删除 操作 的 运行 结果 图 
删除 消息 时 ， 选 择 要 删除 的 消息 列 ， 单 击 “ 删 除 ” 超 链接 ， 弹 出 “确认 删除 ”对 话 框 ， 单 击 “ 确 
定 ”按钮 ， 将 要 删除 消息 的 id 号 传 给 系统 ， 系 统 通过 消息 处 理 函 数 将 消息 删除 。 关 键 代码 如 下 : 
例 程 10 ”代码 位 置 : 光盘 \TMWO3\oaNsxxvd message_chk php 


<?php 
session start(): /开启 session 
include "../inc/chec.php": // 包 含 权 限 检查 文件 


Qs3) 
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include "../conn/conn.php"; 
include "../inc/func.php": 
/删除 数据 ”%/ 

Ssqlstr = "delete from tb_ person where id= ".$_GET[id]: 
Sresult = mysql_query($sqlstr,$conn): 
re_message(Sresult,"./p_manage.php"):; 


/说 宙 实 实 实 灾 闪闪 光 / 


// 包 含 数 据 库 链 接 文 件 
// 包 含 函 数 处 理 文件 


// 消 息 处 理 函 数 


?> 
3.7.4 意见 箱 的 实现 过 程 


国 ”意见 箱 模块 使 用 的 数据 表 : tb lyb 
意见 箱 是 用 户 匿名 发 言 的 一 个 留言 板 。 对 于 用 户 的 留言 ,只 有 人 事 部 的 人 员 才 有 权限 回复 或 删除 ， 
对 于 其 他 部 门 的 员工 则 只 是 可 以 留言 和 查看 , 而 无 法 进行 其 他 操作 。 意见 箱 的 运行 结果 如 图 3.21 所 示 。 
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图 3.21 意见 箱 页 面 的 运行 结果 
在 本 页 面 中 ， 系 统 首先 读 取出 意见 箱 数据 表 (tb_ lyb) 中 的 所 有 数据 ， 并 逐条 显示 ， 在 每 条 显示 记 
录 后 ， 有 一 个 判断 式 ， 判 断 用 户 所 属 部 门 是 否 为 人 事 部 。 如 果 是 人 事 部 成 员 ， 将 显示 “回复 ”和 “ 删 
除 ” 超 链接 ， 如 果 不 是 人 事 部 ， 则 不 显示 这 两 个 操作 。 这 里 是 强制 性 的 ， 也 就 是 说 ， 即 使 是 管理 员 ， 
也 无 法 设置 这 里 的 权限 ， 只 有 开发 人 员 才 可 以 改变 。 如 果 想 使 用 这 种 技术 ， 就 一 定 要 确保 这 项 功能 权 
限 不 会 轻易 的 更 改 。 下 面 给 出 意见 箱 页 面 的 关键 代码 。 
例 程 11 代码 位 置 : 光盘 \TM\03\oa\zytd\lyb.php 


<?php 

session start(): // 开 启 session 支 持 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/chec.php": // 包 含 权 限 检查 文件 
include "../inc/fimc.php": // 包 含 自 定义 函数 文件 
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/显示 意见 箱 信息 ”/ 
$1 sql = "select idl title,] content] time,is reply,r back from tb lyb order by id desc": 
$L rst= mysql query($] sqlLSconn): 


while($1 rows = mysql fetch array($l rst)){ // 开 始 循 环 输出 每 条 记录 
?> 
<t> 
<td width="15%" height="30" align="left" valign="middle" scope="col"> 
<?php echo $1 rows[l time]: ?> 
</td> 
<td align="left" valign="middle" scope="col"> 
主题 ，<?php echo $l_rows[l_title]: ?> 
</td> 
<!-- 判断 登录 用 户 是 否 为 人 事 部 成 员 -> 
<td width="15%" align="center" valign="middle" scope="col"> 
<?php if(($_SESSION[u_depart]) 一 "人 事 部 "){ ?> 
<!-- 当 登录 用 户 是 人 事 部 成 员 时 ， 显 示 可 执行 操作 --> 
<a href="t_back.php?id=<?php echo $1 rows[id]: ?>"> 回 复 </a>&nbsp:&nbsp:<a href="del ly_chk.php?id=<?php 
echo $1_rows[id]: ?>" onclick="return del_ mess0"> 删 除 </a> 
<?php } 
?>&nbsp: 
</td> 
<<!-- --------- 一 -一 --- 一 ----- 一 -一 -一 -一 --- 一 - 一 > 
</tr><tr> <td height="75" colspan="3" align="left" valign="middle" scope="col" > 
<?php echo $1 rows[l_content]; ?> 
</td></tr> 
<2php 
<!-- 当 字段 is_reply 的 值 为 1 时 ， 说 明 有 有 回复 信息 --> 
这 $l rows[is reply] — 1){ 
?> 
<t> 
<td height="25" colspan="3" align="left" valign="middle" headers="50" scope="col" 
‘onmouseover="this.style.backgroundColor=#FFEEBC"" onmouseout="this.style.backgroundColor=""> 
<font color="#FF0000"> 
<!-- 显 示 回 复 信息 --> 
<?php echo $1 rows[r back]: ?> 
</font></td> </t> 
<?php}} 
/| 
?></table> 
<! 一 留言 菜单 > 


下 面 介绍 匿名 留言 和 回复 、 删 除 留言 的 实现 过 程 。 
1. 匿名 留言 


留言 页 面 主要 是 一 个 HTML 表单 ， 因 为 允许 匿名 留言 ， 所 以 表单 中 没有 隐藏 域 ， 该 页 面 中 的 表 间 
元 素 列表 如 表 3.6 所 示 。 


艺 
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表 3.6 留言 页 面 的 菜单 表单 元 素 表 


1 content textarea name="] content" cols="75" rows="10" | 意见 内 容 
Submitl hidden type="reset" value=" 重 置 " “ 重 置 ”按钮 
Submit submit type="submit" value=" 提 交 " onclick="retum add lyb0: 


当 用 户 填写 完 意见 标题 和 意见 内 容 后 ， 单 击 “ 提 交 ” 按 钮 发布 意见 ， 系 统 将 转 到 意见 处 理 页 
(lyb_chk.php) 中 对 意见 进行 处 理 , 如 果 意见 标题 和 意见 内 容 都 不 为 空 , 则 执行 insert 语句 , 向 表 tb_lyb 
中 添加 一 条 新 数据 。 程 序 的 关键 代码 如 下 : 


<?php 
session start(): /开启 session 支 持 
include "../conn/conn.php"; /包含 数 据 库 链接 文件 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 


族 ”判断 输入 是 否 为 空 ， 如 果 为 空 ， 则 返回 ， 如 果 无 错误 ， 则 执行 insert 语 句 添加 新 记录 */ 
这 (trim($_POSTIL title]) := ") and (trim($_POST[L content]) != ""){ 


$l rst =mysql query($] sql,$conn): 


re_message($] rst,"lyb.php?u id=24"): /使 用 自 定义 函数 
了 
} 
else 
echo "<script>alert( 内 容 和 消息 不 允许 为 空 ):history.go(-1):</script>": 
ye 


2. 回复 、 删 除 留言 操作 

在 意见 箱 页 面 中 ， 当 系统 判断 当前 登录 用 户 为 人 事 部 成 员 后 ， 将 显示 留言 信息 后 面 的 、 可 操作 的 
功能 模块 “回复 ”和 “删除 ”。 单 击 相应 留言 主题 后 面 的 “回复 ” 超 链 接 ， 进 入 回复 留言 页 面 。 回 复 
留言 页 面 的 运行 结果 如 图 3.22 所 示 。 


awt en __ OA 


田 作 全 业 信息 rr: 
二 企业 病灶 页 。 公 司 全 认 页 亏 十 乓 < 和 | 


图 3.22 回复 页 面 的 运行 结果 
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回复 留言 页 面 中 的 主要 表单 元 素 如 表 3.7 所 示 。 
表 3.7 留言 回复 表单 的 主要 元 素 表 


tback action="t back chkphp" method="post” 


t title text | value="<?php echo read field($conn."tb lyb"."l title".$_ GET[id]): ?>" | 意见 主题 

r back textarea | name="r back" id="r back" cols="75" rows="10" | 回复 内 容 

id hidden | name="id" value="<?php echo $_ GET[id]: ?>" | 回复 消息 id 
submit2 submit value=" 回 复 " onclick="return re backO:” “回复 ”按钮 


在 回复 留言 页 面 中 ， 当 用 户 输入 回复 信息 后 ， 单 击 “ 回 复 ” 按 钮 ， 提 交 表 单 ， 在 表单 处 理 页 
(t_back_chk.php》 中 系统 将 会 根据 回复 的 信息 更 新 表 记 录 。 关 键 代码 如 下 : 


例 程 12 ”代码 位 置 ; 光盘 \TMWO3\oavzytd\t back chkphp 


<?php 

session start(); /开启 session 支 持 
include "../conn/conn.php": /包含 数据 库 链 接 文件 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php": // 包 含 自 定义 函数 文件 


* 根 据 id， 更 新 数据 库 */ 

$sqlstr = "update tb_lyb setr_ back=".$ POST[r back].",is reply = 1 where id=".$ POST[id]: 
$result = mysql_query($sqlstr,$conn):; 

re_message(S$result,"lyb.php?u id=24"); // 信 息 函 数 


/党 突 突 突 宙 兴 兴 内 突 突 闪闪 兴 兴 闪闪 详实 宙 / 


?> 

对 于 不 需要 再 保留 的 留言 信息 ， 可 将 其 删除 。 单 击 相应 消息 主题 后 面 的 “删除 ” 超 链接 ， 弹 出 “ 确 
认 删 除 ” 提 示 对 话 框 ， 单 击 “ 确 定 ” 按 钮 将 消息 的 id 传 给 系统 ， 系 统 通 过 SQL 语句 将 其 删除 。 程 序 的 
关键 代码 如 下 : 

例 程 13 ”代码 位 置 : 光盘 \TMVWO3\oavzytdvdel ly_chkphp 


<2php 

session start(): // 开 启 session 支 持 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../inc/func.php": // 包 含 自 定义 函数 文件 
$sqlstr = "delete from tb_lyb where id = ".$_GETI[id]: // 删 除 信息 的 sql 语 句 
Sresult = mysql_query($sqlstr.Sconn): // 执 行 语句 
re_message(Sresult."lyb.php?u id=24"): // 返 回 结果 

?> 


3.8 考勤 管理 模块 设计 


考勤 功能 是 办 公 自 动 化 管理 系统 中 每 天 都 要 使 用 到 的 功能 模块 之 一 ， 也 是 比较 重要 的 模块 之 一 。 
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其 除了 包括 正常 的 上 下 班 登记 功能 外 ， 还 要 包括 病 事假 登记 和 加 班 登记 等 特殊 登记 功能 。 管 理 员 通 过 
考勤 记录 实现 对 员工 上 下 班 及 加 班 标准 时 间 的 设置 。 


3.8.1 考勤 管理 模块 概述 
考勤 模块 的 主要 功能 有 : 
上 下 班 登记 : 包括 上 班 登记 和 下 班 登记 。 
回 ” 病 事假 登记 : 包括 病假 登记 和 事假 登记 。 
加 班 登记 包括 加 班 上 班 登记 和 加 班 下 班 登记 。 


回 ”考勤 记录 : 包括 上 下 班 标 准时 间 设置 和 加 班 标准 时 间 设 置 。 
考勤 模块 的 活动 图 如 图 3.23 所 示 。 


下 班 登记 加 班 上 班 登记 加 班 下 班 登记 
让 > else 让 else 让 Oe 让 《<> else 


(un 上 整 点 XA 早退 A 整 点 (a A 整 点 人 早退 人 ss 点 
© LO | Lo 。 


图 3.23 考勤 管理 活动 图 
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3.8.2 ”考勤 管理 模块 技术 分 析 


在 PHP 中 查询 数据 表 中 数据 的 一 般 步 骤 为 : 

(1) 链接 数据 库 〈 链 接 文件 conn.php) 。 

(2) 执行 SQL 语句 ， 返回 查询 结果 集 ($result = mysql_query($sqlstr,$conn)。 

(3) 配合 while 循环 语句 输出 查询 结果 (while($rows = mysql fetch_row($result))) 。 

(4) 关闭 数据 库 。 

在 步骤 (3) 中 ， 如 果 要 取得 某 个 字段 的 值 ， 使 用 $rows[$num] 即 可 。 但 通过 图 3.23， 读 者 可 以 发 
现 ， 从 开始 登记 到 保存 登记 记录 ， 虽 然 只 是 一 个 功能 的 实现 ， 却 要 进行 至 少 3 次 判断 。 如 果 都 用 
“0,1,2…” 做 数组 下 标 ， 过 一 段 时 间 后 ， 根 本 记 不 起 取得 的 数据 是 什么 、 有 什么 作用 。 在 后 期 测试 和 
系统 维护 时 ， 还 要 逐一 地 查找 数据 表 来 比 对 ， 相 当 麻 烦 。 除 了 mysql_fetch_rows0 函 数 外 ，php 还 提供 
了 其 他 相似 的 函数 ， 下 面 就 来 学 习 一 下 。 

1. mysql_fetch_array() 函 数 

mysql_fetch_array0) 函 数 的 用 法 和 mysql_fetch_row0 函 数 的 用 法 十 分 相似 ， 唯 一 不 同 的 是 结果 集 的 
数组 下 标 是 所 查找 的 数据 表 的 字段 值 。 

例如 ， 读 取 数 据 表 tb_register 中 的 “登记 时 间 ” 字 段 ， 代 码 如 下 : 


$str = $rows[r_time]; 

这 样 ， 即 使 时 间 间 隔 比 较 长 ， 甚 至 换 人 开发 ， 也 能 明白 该 变量 的 含义 。 

2. mysql_fetch_object() 函 数 

mysql_fetch_object0 函 数 与 其 他 两 个 函数 的 最 大 不 同 就 在 于 : 获取 的 结果 集 是 以 对 象 的 形式 存储 
的 ， 而 不 是 数组 的 形式 。 也 就 是 说 在 读 取 字段 变量 时 所 使 用 的 格式 不 再 是 $rows[$Snum] 了 ， 代 码 如 下 : 

$str = $rows-->r time; 

下 面 来 看 一 个 实例 ， 分 别 使 用 上 面 3 种 不 同 的 函数 来 获取 数据 表 tb_setup 的 jd 值 ， 代 码 如 下 : 


<2php 
入 链接 数据 库 */ 
S$conn = mysql_connect("localhost"."root"."root"): 
mysql_select_db("db_office".$conn): 
Inysql query("set names gb2312"): 
履 ”创建 和 执行 sql 语 句 */ 
$sqlstr = "select * from tb_setup"; 
Sresult = mysql_query($sqlstr.$conn): 
访 ”使 用 mysql_retch_rows0 函 数 输出 字段 值 */ 
Srows_1 = mysql fetch row(S$result); 
$str = "mysql fetch rows 函 数 : ".$Srows_1[0]."<br>"; 
入 ”使 用 mysql_ retch_array0 函 数 输出 字段 值 */ 
Srows 2 = mysql fetch_array(Sresulb: 
$str .="mysql fetch_array 函 数 : ".Srows_2[id]."<br>": 
广 ”使 用 mysql_retch_object0 函 数 输出 字段 值 */ 
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$rows 3 =Iysql fetch object(Sresulb: 
S$str = "mysql _fetch object 函 数 : ".Srows_3->id."<br>" 
echo "<script>alert(".$str."):</script>”: 


输出 结果 为 : 


Imysql fetch rows 函 数 : 1 
Imysql fetch_array 函 数 : 2 
mysql_fetch_object 函 数 : 3 


3.8.3 ”上 下 班 登记 的 实现 过 程 


国 上 下 班 登记 使 用 的 数据 表 : tb Tegister 

选择 “考勤 管理 ”菜单 中 的 “上 下 班 登记 ”菜单 项 ， 在 主 显示 区 (mainFrame) 内 显示 个 人 上 下 班 
的 登记 记录 和 上 下 班 登记 链接 , 用 户 可 以 在 这 里 进行 登记 和 查找 。 上 下 班 登记 页 面 的 运行 结果 如 图 3.24 
所 示 。 


上 下 三 本 记 
肌 恩人 而 消息 
i EL EL] Bm EK 才 证 
2007-11-17 17:10:00 下 班 早退 无 
ee Ee TH- [ET 证 正点 下 三 一 | 
pe | 2001-11-18 10:00:29 上 五 Ga sdf 
|- © 和 mii 记 | 2007-11-18 10:%:32 上 三 讶 到 df 
t 罗 才 双 记 杂 2007-11-18 10:07:06 二 班 迟到 “sdf 
ER 2007-11-18 10:0:15 TT 早退 sf 
用 天 职员 天 地 


图 3.24 上 下 班 登记 的 运行 结果 


在 上 下 班 登记 页 面 中 ， 通 过 登录 用 户 的 id， 可 从 数据 表 tb_register 中 返回 用 户 的 登记 记录 集 ， 并 
将 其 显示 在 页 面 中 。 在 显示 数据 时 ， 需 要 进行 2 次 判断 : 一 次 判断 用 户 的 登记 类 型 (上班 还 是 下 班 ) ， 
一 次 判断 用 户 的 登记 状态 〈 迟 到 、 早 退 、 正 点 上 、 下 班 )》 。 关 键 代 码 如 下 : 

例 程 14 ”代码 位 置 : 光盘 \TM\03\oa\kqgl\work_note.php 


<?php 
session start(): // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/func.php": // 包 含 自 定义 函数 文件 
?> 
<link href="../css/style.css" rel="stylesheet" /> /引进 外 部 css 样 式 文件 
<script sre="../ijs/client js.js"></script> // 引 进 js 脚 本 文件 
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0 <a href="#" onclick="javascript:open Win=open('p_login.php?r id=<?php echo $ GET[u id]; ?>' ,","width= 
420,height=260,scrollbars=n0");"> 
<?php 
广 ”判断 是 哪 种 功能 类 型 “/ 
@ if($ GET[u id] — 14){ 
echo "上 下 班 登记 "; 
}else if($ GET[u id] — 15){ 
echo " 病 事假 登记 ": 
}else if($ GET[u id] 一 16){ 
echo "加 班 登记 "; 
?></a> 
<?php 
让 ”显示 登记 信息 */ 
$sqlstr = "select id,r_date,r time,r type,r_ state,r remark from tb register wherer id=".$ GET[u id]." and p_id 
"S_SESSION[id]; 
Sresult = mysql query($sqlstr.$conn): 
while($rows = mysql_fetch array(Sresult){ 


外 


?> 
<tr> 
<!-- 输出 登记 日 期 -> 
<td height="25" align="center" valign="middle"><?php echo Srows[r_date]; ?></td> 
<!-- 输出 登记 时 间 --> 
<td height="25" align="center" valign="middle"><?php echo Srows[Tr time]; ?></td> 
<td height="25" align="center" valign="middle"> 
<!-- ”输出 登记 类 型 --> 
<?php 
© switch(Srows[r_type]){ 
case 0: 
echo "下 班 "; 
break: 
Case 1: 
echo "上 班 ": 
break:; 
Case 2: 
echo "加 班 签到 ": 
break: 
Case 3: 
echo "加 班 签 退 "; 
break: 
case 4: 
echo "病假 ": 
break: 
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<td height="25" align="center" valign="middle"> 
<!-- 输出 登记 状态 -一 
<?php 
@ Switch(Srows[r type]){ 
case 1: 
echo ($rows[r_state] 一 0)?2" 正 点 上 班 ":" 迟 到 ": 
break:; 
case 0: 
echo ($rows[r_state] 一 0)?" 早 退 "" 正 点 下 班 "; 
break: 
Case 2: 
echo ($rows[r_state] 一 0)?" 正 点 加 班 ":" 晚 点 加 班 "; 
break: 
Case 3: 
echo ($rows[r_state] 一 0)2" 加 班 早退 ":" 加 班 下 班 ": 
break: 
case 4: 
echo ($rows[r_state] == 0)2" 病 假 ":" 事 假 ": 
break; 


<!-- 输出 备注 -> 
<td height="25" align="center" valign="middle"><?php echo (Srows[r _ remark] ‘= nul)?Srows[r_remark]: 
无 ; ?></td> 
</b> 
<?php }?> 


/这 守 突 罕 突 沉沉 光 闪 闪闪 容光 闪闪 闪闪 内 闪闪 容光 实 突 内 闪光 兴 袖 /| 


人 代码 贴 二 


窗口 事件 ， 打 开 一 个 新 的 窗口 。open 函数 的 基本 格式 为 open(url.name,feature)。 其 中 url 为 要 打开 的 地 址 ，name 为 打开 
窗 体 的 名 字 ，features 为 窗 体 属性 特征 ， 如 宽 、 高 等 。 

@ 根据 u id 来 判断 是 哪 种 功能 类 型 ， 也 就 是 参数 U id 的 值 。 分 3 种 情况 : 上 下 班 登记 、 病 事假 登记 和 加 班 登记 。 

@ 判断 登记 类 型 ， 一 共 分 6 种 情况 : 上 班 、 下 班 、 病 假 、 事 假 、 加 班 答 到 和 加 班 签 退 。 

@ 判断 状态 类 型 ， 一 共 分 10 种 情况 : 上 班 迟 到 、 上 班 早退 、 正 点 上 班 、 正 点 下 班 、 正 点 加 班 、 加 班 下 班 、 加 班 晚 
点 、 加 班 早退 、 病 假 和 事假 。 


当 单 击 “ 上 下 班 登记 ” 超 链接 时 ， 就 会 弹出 登记 页 面 ， 用 户 就 可 以 进行 上 下 班 的 登记 操作 了 。 在 
“登记 类 型 ”下 拉 列 表 框 中 选择 登记 类 型 ， 如 果 是 迟到 或 早退 等 特殊 情况 ， 需 要 在 “备注 ”中 说 明 原 
因 ， 最 后 单 击 “ 登 记 ” 按 钮 进行 登记 ， 如 果 登 记 成 功 ， 将 显示 提示 对 话 框 。 登 记 页 面 的 运行 结果 如 
图 3.25 所 示 。 

在 登记 界面 中 ， 主 要 有 3 个 表单 元 素 和 1 个 隐藏 表单 ， 如 表 3.8 所 示 。 
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Ts | 
姓名 : 区 
ex 型: [上 得 可 
主 一 一 
备注 : 
| 习 
图 3.25 上 下 班 登 记 的 运行 结果 
表 3.8 上 下 班 登记 的 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含义 
Pl form method="post" action="p_login_chk.php" 登记 表单 
u name text value="<?php echo $_SESSION[u_name]: ?>" readonly="readonly" 登记 人 


<2php 这 S_GET[r id] 一 14){f?> 
<option value="1"> 上 班 </option> 
<option value="0"> 下 班 </option> 
<?php }else if($_GET[r id] 一 15){?> 
"4"> 病 假 </option> 


utype select > 事假 <Joption> 登记 类 型 
<?php }else if($ GETIr id] — 16){7> 
<option value="2"> 加 班 签到 </option> 
<option value="3"> 加 班 签 退 </option> 
<?php }?> 
r remark textarea name="r_remark" rows="5" 备注 
rid hidden value="<?php echo$_ GET[r id]: ?>" 功能 id 
Submit2 submit Value=" 回 复 " onclick="return re back0:" “登记 ”按钮 


选择 相应 的 登记 类 型 ,如 上 班 ， 提 交 表单 后 ， 系 统 就 会 跳 到 处 理 页 (P_login_chk.php) 处 理 。 在 处 
理 页 中 ， 首 先 判断 登记 类 型 (上班 、 下 班 ) ， 然 后 根据 登记 类 型 取得 相对 应 的 标准 时 间 ， 通 过 两 个 时 
间 的 对 比 ， 得 到 登记 状态 〈 迟 到 、 早 退 、 正 点 上 下 班 ) ， 最 后 执行 insert 语句， 添加 新 记录 。 程 序 的 关 
键 代码 如 下 : 


例 程 15 ”代码 位 置 : 光盘 \TM\03\oa\kqgl\p_login_chk.php 


<2php 

Session start(): // 开 启 session 支 持 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/chec.php": // 包 含 权限 检查 文件 


具 1 代表 迟到 0 代表 正点 登记 */ 

上 # ”判断 功能 类 型 ， 如 上 下 班 登记 、 加 班 登 记 。*/ 
这 (Sr id — "14") or (sr id— "16"){ 

语 判断 登记 类 型 ， 如 上 班 、 下 班 等 */ 
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if($u type 一 0) 

S$t_sql = "select * from tb_setup where id = 2"; // 取 得 下 班 标 准时 间 
else if($u type — 1) 

S$t_sql = "select * from tb_setup where id= 1"; // 取 得 上 班 标准 时 间 
else if($u type — 2) 

St_sql = "select * from tb_setup where id= 3": // 取 得 加 班 上 班 时 间 
else if($u type — 3) 

St_sql = "select * from tb_setup where id = 4": // 取 得 加 班 下 班 时 间 

/说 认 兴 突 宙 突 闪 突 闪闪 闪闪 闪闪 闪闪 闪闪 突 兴 闪闪 闪闪 认 兴 突 闪 闪闪 突 兴 认 兴 闪闪 次 这 闪闪 次 突 闪 诡 闪闪 闪 突 次 安 闪 交 交 实 家 

St rst = mysql query($t_sql,$conn); 
St_rows = mysql fetch row(S$t_rst); 


$s_time = $t_rows[2]: // 规 定 的 标准 时 间 
9 Snow_time = date("H:i:s"):; // 当 前 的 登记 时 间 
© $1_sql = "insert into tb_register (r_date.r time,r type.r_state,r remark,r idp id) 


values("".date("Y-m-d").",".date("H:i:s").",".$_POST[u typel].".".((strtotime(Snow_time) - strtotime($s_time) > 
0)21:0)."".$_POST[r remark].".,".$ POST[r id].".".$_SESSION[id].")": 
}else if($r id— "15") 
$1_sql = "insert into tb_register (r_date,r_time,r type,r_state,r remark,r id,p id) 
Values(".date("Y-m-d").",".date("H:i:s").",".$_ POST[u type].".",".$_POST[r_ remark].".".$_POST[r id].",".$_SESSION[id]." 
多 


$l rst=Imysql_ query($] sql,$conn): 
/* ”登记 完成 后 ， 关 闭 窗口 。 */ 
if($] rst) 
echo "<script>alert(' 登 记 完成 ');window.closeO:</script>"; 
else 
echo "<script>alert( 错 误 "):history.go(-1):</script>"; 


/这 素 率 率 素 来 六 素来 闪 素 来 率 率 率 率 六 认 率 率 六 让/ 


?> 


< 八代 码 贴 寺 

@ date0 时 间 函 数 可 以 通过 不 同 的 组 合 返回 不 同 的 时 间 样 式 。 如 程序 中 的 date("H:i:s")， 还 可 以 加 上 年 、 月 、 日、 星 
期 等 等 ， 如 date("Y-m-d His 1")。 

四 ((strtotime($now_time) - strtotime($s_time) > 0)?1:0): 根据 以 上 内 容 得 到 的 SQL 语句 ， 其 中 strtotime 函数 是 PHP 
的 一 个 时 间 函 数 ， 作 用 是 将 一 个 日 期 时 间 描述 解析 为 一 个 unix 时 间 蕉 ， 如 strtotime("2007-07-11 13:49:29") 输出 的 结果 
就 是 1184132969， 这 时 对 两 个 时 间 进 行 计 算 ， 就 可 以 得 到 时 间 的 大 小 关系 了 。 


3.8.4 设置 时 间 的 实现 过 程 


国 ”考勤 管理 使 用 的 数据 表 : tb register、tb_setup 

考勤 记录 页 面 的 功能 主要 分 为 两 部 分 ， 一 是 显示 所 有 员工 的 考勤 信息 ， 二 是 设置 标准 时 间 。 显 示 
所 有 员工 的 考勤 信息 的 实现 过 程 和 3.8.3 节 中 的 实现 方式 相 类 似 ， 唯 一 不 同 的 就 是 包括 了 全 体 职员 ， 只 
要 略微 改动 SQL 语句 即 可 。 

下 面 主要 来 看 一 下 设置 标准 时 间 页 面 。 在 “考勤 记录 ”页 面 中 单 击 “ 设 置 时 间 ” 超 链接 ， 进 入 设 
置 时 间 页 面 ， 设 置 时 间 页 面 的 运行 结果 如 图 3.26 所 示 。 
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| HN, | TREE Nm | ?PE ， 
[gaa [rer 0 rr 
| | 是 | 本 | 到 | 


图 3.26 设置 时 间 页 面 的 运行 结果 
在 本 页 面 中 ， 从 tb_setup 表 中 读 取 预定 义 的 标准 时 间 。 有 4 个 标准 时 间 ， 分 别 是 “上 班 签到 ”、 
“下 班 签 退 ”、“ 加 班 签到 ”和 “加 班 签 退 ”。 程 序 的 关键 代码 如 下 : 
例 程 16 ”代码 位 置 : 光盘 \TM\03\oa\kqgl\set.time.php 


<2php 
Session _start(: /开启 session 支 持 
include "inc/chec.php": // 包 含 权限 检测 文件 
include "../conn/conn.php"; // 包 含 数 据 库 链 接 文 件 
> 
<link href="../css/style.css" rel="stylesheet" /> /引入 css 外 部 链接 文件 
<script sre="../js/client js.js"></script> /引入 js 脚本 文件 


<form name='"forml" method="post" action="set_time_chkphp'"> 
<table border="1" cellspacing="0" cellpadding="0" background="../images/bg.jpe”"> 
<t> 
<td width="150" height="25" align="center" valign="middle"> 上 班 签到 </td> 
<td width="150" height="25" align="center" valign="middle"> 下 班 签 退 </td> 
<td width="150" height="25" align="center" valign="middle"> 加 班 签到 </td> 
<td width="150" height="25" align="center" valign="middle"> 加 班 签 退 </td> 


</t> 
<?php 
S$sqlstr = "select * from tb_setup": // 提 取 表 中 的 标准 时 间 
Sresult = mysql_query($sqlstr, $conn); // 返 回 结果 集 
Snum = 0: /设置 序号 
?> 
<t> 


<?php while($rows = mysql_fetch row($resulD){ ?> 
<td height="30" align="center" valign="middle"> 


0 <input type="text" name="] time<?php echo Snum;?>" value="<?php echo $rows[2]: ?>" size=15></td> 
<?php Snum++:}?> 
</u><tu> 
@ <td height="30" align="center" valign="middle"><input type="submit" name="u_logo" value=" 设 置 "></td> 


<td height="30" align="center" valign="middle"><input type="submit" name="d_logo" value=" 设 置 "></td> 

<td height="30" align="center" valign="middle"><input type="submit" name="a_logo" value=" 设 置 "></td> 

<td height="30" align="center" valign="middle"><input type="submit" name="q_logo" value=" 设 置 "></td> 
</a> 
</table> 
</form> 


4 代码 贴 十 
@ 变量 Snum 是 为 了 在 while 循环 语句 中 输出 不 同 的 表单 元 素 的 名 称 ， 每 次 循环 Snum 自 加 1。 
四 生成 4 个 设置 时 间 的 按钮 ， 在 处 理 页 中 ， 根 据 不 同 的 名 字 做 不 同 的 处 理 。 


当 调整 了 其 中 的 一 项 后 ， 单 击 “ 设 置 ”按钮 ， 就 进入 到 处 理 页， 在 处 理 页 中 ， 会 根据 按钮 的 名 称 
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和 “text” 元 素 的 value 值 来 进行 相关 的 操作 。 程 序 的 关键 代码 如 下 : 
例 程 17 代码 位 置 光盘 \TM\03\oa\kqgl\set.time_chkphp 


<?php 
session start(); /开启 session 设 置 
include "inc/chec .php": /包含 权限 检查 文件 
include "../conn/conn .php": /包含 数 据 库 链接 文件 


证 根据 名 称 判断 事件 是 哪个 按钮 产生 的 ， 并 生成 不 同 的 SQL 语句 并 
ifisset($_ POST[u_ logo])){ 


$sqlstr = "update tb_setup setl time =".$ POST[L time0]. "where id = 1 /设置 上 班 时 间 
}else if(isset($ POST[d logo])){ 

$sqlstr = "update tb_setup setl_ time =".$ POST[] timel]. " where id = 2"; /设置 下 班 时 间 
}else ifisset($_ POST[a logo])){ 

S$sqlstr = "update tb_setup set | time =".$_ POST[L time2]. "where id = 3": /设置 加 班 上 班 时 间 
}else if(isset($ POST[q logo])){ 

S$sqlstr = "update tb_setup setl time =".$ POST[L time3]. " where id = 4": /设置 加 班 下 班 时 间 


上 
else 

echo "<script>alert(' 非 法 登录 "):window.close():</script>"; 
让 帮 员 间 让 浊 检 检 宁 兴 汪 市 办 办 宙 检 本 订 兴 业 订 中 中 胃 订 水 页 只 烛 员 只 间 蓝 认 刘 杰 员 虽 中 市 烛 浊 宙 检 水 中 业 非 中 村 四 南 牛 
Sresult = mysql_query($sqlstr,$conn): 
if($result) 

echo "<script>alert(' 设 置 成 功 ):window.closeO:</script>": 
else 

echo "<script>alert( 设 置 失 败 "):window.close():</script>"; 


3.9 后台 首页 设计 


由 于 后 台 管 理 主要 是 针对 管理 员 开 放 的 ， 普 通用 户 无 法 访问 到 这 里 ， 所 以 在 页 面 风格 的 设计 上 相 
对 要 简单 得 多 。 不 需要 艳丽 的 图 片 、 极 炫 的 特效 ， 干净 、 整 洁 ， 给 人 一 种 朴实 无 华 的 感觉 是 最 好 的 。 


3.9.1 后 台 首 页 概述 


如 果 说 前 台 是 系统 的 外 衣 ， 那 么 后 台 就 是 系统 的 灵魂 了 。 数 据 更 新 、 系 统 升级 ， 甚 至 是 安全 隐患 ， 
都 源 于 后 台 的 设计 是 否 完 善 ， 功 能 是 否 强 大 。 可 以 说 ， 后 台 越 完美 ， 系 统 拥有 的 外 衣 才 会 越 华丽 ， 人 
们 的 目光 才 会 被 吸引 。 

办 公 自 动 化 管理 系统 的 后 台 首 页 主要 包括 的 功能 如 下 : 

网 站 首部 导航 栏 : 包括 当前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 

网 站 左 侧 导航 栏 : 包括 各 个 管理 模块 及 分 类 。 

部 门 管理 模块 : 主要 用 于 对 部 门 信息 的 操作 ， 包 括 添加 、 修 改 、 删 除 和 查看 功能 。 
职员 管理 模块 : 主要 用 于 对 职员 信息 的 统一 管理 ， 包 括 添加 、 修 改 、 删 除 和 查找 职员 。 
权限 管理 模块 : 主要 用 于 对 部 门 、 职 员 和 用 户 组 的 功能 权限 分 配 。 


回回 罗 加 加 
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回 “系统 管理 模块 : 维护 系统 日 志 、 修 改 管理 员 密码 和 系统 数据 的 备份 。 
下 面 看 一 下 本 案例 中 提供 的 后 台 首 页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\03\admin\admin_ 
main.php， 如 图 3.27 所 示 。 


图 3.27 办 公 自 动 化 管理 系统 后 台 首页 


3.9.2 ”后台 首页 技术 分 析 


后 台 页 面 和 前 台 页 面 的 布局 是 相似 的 ， 都 是 采用 “上 方 固定 ， 左 侧 嵌 套 ” 的 框架 结构 。 在 上 方 框 
架 (topFrame) 显示 banner 和 首部 导航 ， 在 左 侧 框 架 〈leftFrame) 中 显示 功能 列表 。 中 间 的 显示 区 
(manFrame) 显示 各 级 窗 体 。 


3.9.3 后台 首 页 的 实现 过 程 


国 ”后台 首页 使 用 的 数据 表 : tb_controller 
下 面 来 看 一 下 后 台 页 面 中 框架 布局 的 应 用 ， 程 序 代码 如 下 : 


例 程 18 代码 位 置 : 光盘 \TM\03\oa\admin\admin_main.php 
<!-- 定 义 顶 部 框架 和 框架 内 显示 的 内 容 --> 
<frameset rows—"88,*" cols="*" frameborder="no" border="1" framespacing="0"> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
<!-- 定 义 嵌 套 的 左 侧 框架 和 框架 内 显示 的 内 容 --> 
<frameset rows="*" cols="216,*" framespacing="0" frameborder="no" border="1"> 
<frame src="left.php" name="leftFrame" scrolling="No" noresize="noresize" id="leftFrame" title="leftFrame" /> 
<!-- 定 义 主 框架 内 容 --> 
<frame src="main.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
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<!-- 当 浏览 器 不 支持 框架 时 所 显示 的 内 容 -> 
<noframes><body> 

</body> 

</noframes></html> 


3.10 ”部 门 管理 模块 设计 


一 个 成 熟 、 稳 定 的 企业 一 定 要 有 一 个 合理 的 部 门 结构 ， 包 括 每 个 部 门 需要 做 什么 工作 ， 部 门 之 间 
的 协调 关系 和 部 门 之 间 的 上 下 级 关系 等 ， 都 要 明确 下 来 ， 这 样 才能 各 尽 其 职 ， 不 会 造成 资源 浪费 ， 提 
高 企业 整体 的 效率 。 


3.10.1 部 门 管理 模块 概述 


部 门 管理 模块 包括 两 大 部 分 : 

1. 查看 部 门 

查看 各 个 部 门 名 称 和 部 门 之 间 的 上 下 级 关系 ， 还 可 以 对 部 门 的 信息 和 级 别 进行 修改 ， 如 果 是 最 底 
层 的 部 门 ， 则 可 以 对 其 进行 删除 操作 。 

2， 添 加 部 门 

可 以 添加 新 的 部 门 ， 包 括 部 门 名 称 、 上 级 部 门 和 部 门 备注 。 

部 门 管理 框架 如 图 3.28 所 示 。 


部 门 管理 
退 到 登录 界面 
否 


判断 是 否 为 管理 员 


[| mi | 


[[ 看 wn | 


判断 部 门 名 称 


添加 部 门 


修改 部 门 删除 部 门 


删除 失败 


删除 成 功 


图 3.28 部门 管理 操作 流程 图 
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3.10.2 ”部 门 管理 模块 技术 分 析 


部 门 管理 中 的 技术 难点 是 : 如 何 实 现 查 看 部 门 模块 中 的 动态 显示 部 门 结构 的 功能 。 这 里 用 
javascript+ 递 归 函 数 来 实现 。 下 面 详细 讲解 。 

当 用 户 选 择 “查看 部 门 ” 命 令 时 ， 系 统 开始 查找 tb_depart 数据 库 。 首 先 找到 根部 门 的 信息 列表 ， 
并 判断 第 1 个 根部 门 是 否 有 下 级 部 门 。 如 果 没 有 ， 则 直接 输出 部 门 名 称 ， 执 行 修改 和 删除 操作 ， 然 后 
查找 下 一 个 根部 门 ， 如 果 有 下 级 部 门 ， 则 调用 自 定义 函数 list menu($num.$wid.&$m)。 其 中 ， 

$num: 为 上 级 部 门 id 号 。 

回 $wid: 为 表格 宽度 。 

$m: 为 表单 元 素 id 变量 值 。 

在 这 个 函数 中 ， 执 行 步骤 如 下 : 

(1) 查找 上 级 部 门 为 $num 的 部 门 列表 。 

(2) 判断 第 1 个 部 门 是 否 有 下 级 部 门 。 

(3) 如 果 没 有 ， 则 输出 部 门 名 称 ， 修 改 和 删除 操作 ， 然 后 查找 下 一 个 同 级 部 门 。 

(4) 如 果 有 下 级 部 门 ， 则 再 次 调用 list menu($num.$wid,&$nm) 函 数 。 

(5) 重复 步骤 (1) 一 步骤 (4) 。 

这 时 ， 相 信 读 者 已 经 看 出 list menu0 正 是 一 个 递归 函数 。 递 归 函 数 就 是 在 当 满足 一 定 的 条 件 时 ， 
函数 自己 调用 自己 ， 经 过 层 层 调用 ， 最 终 达 到 解决 问题 的 目的 。list_ menu0 函 数 的 代码 如 下 : 

例 程 19 代码 位 置 : 光盘 \TMW3voavadmin\incfunc.php 


<?php 

function list_ menu($num,$wid,&$Sm){ 
入 ”连接 数据 库 */ 
S$conn = mysql_connect("localhost"."root"."root"): 
mysql_select_db("db_office", $conn); 
mysql_query("set names gb2312"): 
/ 刘 让 中 训 半 训 训 来 宴 事 事 宴 率 率 字 率 事 / 
族 ”查询 同 级 部 门 */ 
S$sqlstr = "select * from tb_depart where up_depart= ".Snum: 
Sresult = mysql_query($sqlstr,$conn): 
ended h 

?> 
<!-- 隐藏 域 --> 

<tr id="OpenMe<?php echo $m: ?>" style= "display:none:"> 

<td> <table width="<2php echo $wid: ?>%" border="0" cellspacing="0" cellpadding="0"> 
<t> 
<td height="25" align="center"><table width="<?php echo ($wid -3): ?>%" border="0" cellspacing="0" 
cellpadding="0"> 


hp 
请 ”循环 输出 同 级 部 门 */ 
while($rows = mysql fetch row(S$result){ 
请 ”查看 下 属 部 门 。 */ 
$sqlstrl = "select * from tb_depart where up_depart = ".Srows[0]: 
Sresult]l = mysql query($sqlstr1.$conn): 
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Snu =mysql num rows($result]); 
入 。 如 果 当 前 部 门 没 有 下 属 部 门 时 */ 


这 !$no{ 
$m += 1; 
?> 
0 <tr onMouseOver="this.style.background='#96F7F4'" onMouseOut="this.style.background=""> 
<td width="100%" align= "left">&nbsp:&nbsp:&nbsp:<img src="../images/folder.gif" width="16" 
height="16" border="0"><?php echo $rows[1] 2>- 一 -一 一 -一 —--<a href="edit_depart.php?id=<?php echo $rows[0]: ?>"> 修 改 


</a>||<a href="del_depart_chk.php?id=<?php echo $rows[0]: ?>" onClick="return cfim():"> 删 除 </a></td></tr> 
<?php 


} 
入 ”当前 部 门 有 下 属 部 门 时  */ 
else{ 
Sm+= 1; 
?> 
<!- ”鼠标 划 过 样式 -> 
<tr onMouseOver= "this.style.background='#96F7F4" onMouseOut="this.style.background=""> 
© <td>&nbsp:<a href—"Javascript:ShowMe(img<?php echo Sm; ?>,OpenMe<?php echo Sm; ?>)"><img 
src="../Images/jia.gif" border="0" alt=" 展 开 " id="img<?php echo Sm; ?>"><?php echo 
Srows[1];?></a>-- 一 -一 -一 -一 -一 - <a href="edit_depart.php?id=<?php echo Srows[0]: ?>"> 修 改 </a>|| 删 除 </td></tr> 
<2php 
© list menu(Srows[0],Swid,Sm); 
} 
?> 


</table></td></tr></table></td></tr> 
<?php}?> 
< 代码 贴 十 
@ onMouseOver…onMouseOut: onMouseOver 表示 鼠标 指针 停留 在 当前 元 素 上 时 的 状态 ，onMouseOnut 事件 表示 鼠 
标 指针 离开 当前 元 素 时 的 状态 。 本 行 用 不 同 的 颜色 表示 两 种 不 同 的 事件 。 
@ 当前 部 门 有 下 属 部 门 时 ， 单 击 图 片 或 文字 链接 ， 触 发 一 个 js 事件， 显示 隐藏 域 。 
四 调用 自身 函数 。 


在 这 个 模块 中 , 同时 使 用 了 另 一 个 自 定义 函数 isbool($id,$dname), 该 函数 将 $dname 参数 和 tb_depart 
表 中 的 部 门 名 称 一 一 比较 ， 如 果 重 复 ， 则 返回 tue， 和 否则 返回 false。$id 是 在 修改 部 门 时 ， 先 去 掉 要 修 
改 的 部 门 的 名 称 再 进行 比较 。 在 信息 处 理 页 中 ,根据 返回 的 值 进行 操作 。isbool0 函 数 的 程序 代码 如 下 : 


例 程 20 ”代码 位 置 : 光盘 \TM\03\oa\admin\inc\func.php 
function isbool($id.Sdname){ 
广 ”连接 数据 库 */ 
S$conn = mysql_connect("localhost"."root"."root"): 
mysql_select_db("db_office".$conn); 
mysql_query("set names gb2312"): 


/ 实 兴 认 突 实 突 灾 闪 闪闪 实 实 突 交 次 实 窑 / 


S$sqlstr = "select * from tb_depart where id (= Sid d_name ='$dname": /创建 查询 SQL 语句 


Sresult = mysql query($sqlstr.$conn): // 执 行 SQL 语 句 
这 mysql num rows($result)>0) // 判 断 是 否 有 记录 集 
Sisbool = true: // 如 果 有 ，Sisbool= true 
else 
Sisbool = false: /否则 Sisbool = false 
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Tetum $isbool: // 返 回 $isbool 
} 


3.10.3 ”部 门 查看 的 实现 过 程 


国 ”部 门 查看 模块 使 用 的 数据 表 : tb_depart 
通过 选择 导航 栏 中 的 “部 门 管理 ”/“ 查 看 部 门 ”菜单 项 ， 进 入 部 门 查看 页 面 。 在 部 门 查看 页 面 中 
显示 部 门 设 置 及 架构 情况 ， 部 门 查看 页 面 的 运行 结果 如 图 3.29 所 示 。 


办 公 自动 化 


dd 
田 er 
"人 下 部 一 一 一 一 一 一 修改 | | 抽 
三 二 中 量 音 一 一 一 一直 改 | 对 
三 硬 妈 寺 刘 一 一 性 下 || 生 了 


图 3.29 部 门 查看 页 面 的 运行 结果 
在 部 门 查看 页 面 中 ， 使 用 到 了 递归 函数 。 首 先 查 找 根部 门 〈 根 部 门 的 条 件 是 top_depart = 0) ， 并 
返回 根部 门 的 结果 集 , 接着 使 用 while 循环 输出 所 有 根部 门 ， 每 输出 一 个 根部 门 时 ， 都 要 查看 当前 部 门 
是 否 有 子 部 门 (下属 部 门 》。 如 果 没 有 子 部 门 , 那么 直接 输出 部 门 名 称 和 对 部 门 的 操作 (修改 和 删除 》; 
如 果 根 部 门 有 子 部 门 ， 则 调用 递归 函数 (ist_menu()) 。 该 页 面 的 程序 代码 如 下 : 
例 程 21 ”代码 位 置 ， 光盘 \TM\03\oa\admin\bmgl\show_depart.php 


<2php 
session_start(): /开启 session 
include "../inc/chec.php": // 包 含 权 限 检查 文件 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/func.php": // 包 含 自 定义 函数 文件 
?> 
<script src="../js/admin jsjs"></script> // 引 入 js 脚本 文件 
<link href="../css/style.css" rel="stylesheet"> // 引 入 外 部 css 样 式 
<body onLoad="ShowMe(img1.OpenMel)"> /onLoad 事 件 ， 表 示 当 页 面 被 载 入 时 就 调用 ShowMe() 方 法 


<!-- 查 看 部 门 信息 --> 
<table width="100%" height="25" border="0" cellpadding="0" cellspacing="0"> 
<?php 

广 ”找到 根部 门 */ 

S$sqlstr = "select * from tb_depart where top_depart = 0"; 

Sresult = mysql query($sqlstr.$conn): 

庆 ”隐藏 域 ia 号 */ 

$m=1: 

人 ”循环 输出 所 有 根部 门 */ 

while($rows = mysql fetch row($result)){ 
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S$wid = 100: 
六。 查看 下 属 部 门 
$sqlstrl = "select * from tb_depart where up_depart = ".Srows[0]: 
Sresult]l = mysql query($sqlstrl,$conn): 
Snu = mysql num rows($result]); 
广 。 如果 当前 部 门 没有 下 属 部 门 时 */ 
这 !$nu){ 
?> 


<tr onMouseOver="this.style.backeround="#96F7F4" onMouseOut="this.style background= "> 
<td>&nbsp:&nbsp:&nbsp:&nbsp:<a hre 全 "<?php echo SPHP_ SELF: ?>"><img src="../Images/folder.gif" 
border="0" alt=" 展 开 " id="img<?php echo $m: ?>"><?php echo Srows[1]; ?></a>-- 一 ---- 一 -一 ---- -<a 
href="edit_depart.php?id=<?php echo Srows[0]; ?>'"> 修 改 </a>||<a href="del_depart_chk.php?id=<?php echo $rows[0] ?>" 
onClick="retum cfim0:"> 删 除 </a></td> 
</t> 


<?php 
店 ”如果 当前 部 门 有 下 属 部 门 时 。 */ 
}else{ 
?> 
<tr onMouseOver="this.style.background='#96F7F4"" onMouseOut="this.style.background=""> 
<td><a href—"Javascript:ShowMe(img<?php echo Sm; ?>,OpenMe<?php echo Sm; ?>)"><img 
src="../Images/jia.gif" border="0" alt=" 展 开 " id="img<?php echo $m: ?>"><?php echo $rows[1]:; ?></a>---—----------- <a 
href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>|| 删 除 </td> 
</t> 
<?php 
人 * ”输出 同 级 部 门 ， 调 用 递归 函数 */ 
list_ menu($rows[0],$wid,$m); 
Sm+= 1; 
/* 循环 结束 ”*/ 


?> 
1， 修 改 部 门 


在 查看 部 门 页 面 中 ， 可 以 对 部 门 的 名 称 、 上 级 部 门 及 主要 职责 等 相关 信息 进行 修改 。 单 击 要 修改 
的 部 门 名 称 后 面 的 “修改 ” 超 链接 ， 进 入 修改 部 门 页 面 。 修 改 页 面 的 运行 结果 如 图 3.30 所 示 。 


部 门 名 称 : RH 二 
上 地名 门 : | 无 号 


计 前 台 页 面 ， 美工 、photoshop、flash 三 


习 
sse] | 


图 3.30 修改 部 门 页 面 的 运行 结果 
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修改 部 门 页 面 (edit departphp) 中 的 主要 表单 元 素 如 表 3.9 所 示 。 


表 3.9 修改 部 门 页 面 中 的 表单 元 素 
元 素 类 型 重要 属性 含义 
Text size="20" value="<?php echo Srows[d name] ?> 部 门 名 称 
<?php 
while($tmprows = mysql_fetch_array($tmpresult){ 
if($tmprows[id] 一 $rows[top_depart) { 


名 称 


d name 


echo "<option value=". $tmprows[id]." 


uid select selected>".$tmprows[d_name]."</option>"; 列 出 所 有 部 门 
jelse 
echo "<option value=". $tmprows[id].">".$tmprows[d_name]." 
</option>"; 
1}?> 
Temark textarea <?php echo Srows[remark] ?> 备注 信息 
id hidden value="<?php echo $rows[id] ?>" 修改 部 门 的 id 
Submit Submit name="Submit" value=" 修 改 " onclick="return a_checkO:;" “修改 ”按钮 


当 管 理 员 修改 完 相应 的 信息 后 单 击 “ 修 改 ” 按 钮 提交 修改 信息 ， 系 统 转 到 处 理 页 (edit_depart_ 
chk.php) ， 首 先 判 断 输 入 的 部 门 名 称 是 否 重复 。 如 果 重 复 ， 则 返回 ， 如 果 不 重复 ， 则 继续 执行 ， 确 定 
修改 后 的 上 级 部 门 和 根部 门 。 关 键 代码 如 下 : 

例 程 22 ”代码 位 置 ， 光盘 \TM\03\oa\admin\bmgl\edit_depart_chk.php 


<? 
session start(); /开启 session 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/func.php"; // 包 含 数据 库 链 接 文件 


/* ”判断 输入 部 门 名 称 是 否 重复 。 */ 

iftisbool($_ POST[id].$_POST[d_name]){ 
echo "<script>alert( 名 称 已 存在 ， 请 重新 输入 !!"):history.go(-1):</script>"; 
exitO: 


广 ”修改 部 门 ， 确 定 上 级 部 门 和 根部 门 */ 
这 $_POST[u id] := 0){ 
证 ”查找 上 级 部 门 的 根部 门 。 */ 
$sqlstr = "select top_depart from tb_depart where id=".$ POST[u id]: 
Sresult = mysql query($sqlstr.Sconn): 
Srows =Iysql fetch array($result); 
启 ”如 果 存 在 上 级 部 门 ， 就 取得 上 级 部 门 的 根部 门 。*/ 
if ($rows[top_depart] != 0) 
Stop_depart = $rows[top_depart]: 
启 ”如 果 不 存在 ， 将 上 级 部 门 作为 根部 门 。*/ 
else 
Stop_depart = $_POST[u id]: 
}else 
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卢 如 果 没 有 上 级 部 门 ， 那 么 就 将 自身 定 为 根部 门 。 */ 
Stop_depart = 0: 
店 ”创建 、 执 行 修改 部 门 的 SQL 语 名 。 */ 
Ssqlstr = "update tb_depart set d name =".$ POST[d namel].".top depart=".$top depart.", up_depart= 
".$_ POST[u id].", remark =".$ POST[remark]." where id = ".$ POSTI[id]: 
Sresult = mysql_query($sqlstr,$conn): 
re_message(S$result,"show_depart.php"); 
> 
2. 删除 部 门 
如 果 要 取消 某 个 部 门 ， 可 以 将 该 部 门 删除 。 在 删除 部 门 时 ， 只 能 对 最 底层 的 部 门 进行 删除 ， 如 果 
部 门下 面 有 子 部 门 ， 其 部 门 后 面 的 “删除 ” 超 链接 不 可 用 。 删 除 功 能 的 实现 相对 比较 简单 一 点 ， 单 击 
“删除 ” 超 链 接 后 ， 系 统 提示 “是 否 确认 删除 部 门 ”， 如 果 单 击 “ 确 认 ” 按 钮 ， 系 统 将 转 到 处 理 页 
(del_depart_chk.php) 进行 删除 处 理 。 程 序 代 码 如 下 : 


例 程 23 ”代码 位 置 : 光盘 \TM\03\oa\admin\bmgl\del_depart_chk.php 


<?php 

session start(); /开启 session 

include "../inc/chec.php"; // 包 含 权限 查看 文件 
include "../conn/conn.php": // 包 含 数 据 库 链 接 文 件 
include "../inc/func.php": // 包 含 自 定义 文件 
$sqlstr = "delete from tb_depart where id =".$_GETI[id]: // 删 除 SQL 语 句 

Sresult = mysql_query($sqlstr, Sconn): // 执 行 SQL 语 句 
re_message(Sresult,"show_depart.php"): // 返 回 最 后 结果 

?> 


3.10.4 部 门 添加 的 实现 过 程 


用 户 选择 导航 栏 中 的 “部 门 管理 ”/“ 添 加 部 门 ” 菜 单项 , 可 以 进入 添加 部 门 页 面 (add_departphp ) ， 
该 页 面 用 于 设置 新 添加 部 门 的 相关 信息 ， 包 括 部 门 名 称 、 上 级 部 门 和 备注 。 添 加 部 门 页 面 的 运行 结果 
如 图 3.31 所 示 。 


办 公 自 动 化 所 


人 人 事 苞 理 、 币 度 管 理 、 保 障 管 理 、 合 同 管 三 
区， 
到 


图 3.31 添加 部 门 页 面 的 运行 结果 
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添加 部 门 页 面 (add_depart) 主要 包含 一 个 添加 部 门 表单 , 表单 元 素 的 名 称 、 说 明 及 属性 值 如 表 3.10 
所 示 。 


表 3.10 ”添加 部 门 页 面 的 表单 元 素 


重要 属性 含 义 
d_name name="d_name" id="d_name" size="20" 部 门 名 称 
<2php 
hile($rows = 1 fetch Sresultl 
让 明 Sia We sa et OC) 列 出 所 有 部 门 


echo "<option value=". $rows[0].">".$rows[1]."</option>"; 


?> 


备注 信息 
“添加 ”按钮 


textarea | name="remark" cols="40" rows="5" id="remark" 


submit name="Submit" value=" 添 加 ”onclick="retum a_check0:" 


当 管理 员 提交 表单 后 ， 进 入 处 理 页 (add_depart_chk.php) ， 进 行 添加 操作 。 在 处 理 页 中 ， 首 先 判 
断 输 入 的 部 门 名 称 是 否 重复 ， 如 果 名 称 符合 要 求 ， 则 进一步 确定 上 级 部 门 和 根部 门 ， 最 后 执行 insert 
语句 进行 添加 操作 。 处 理 页 的 代码 如 下 : 

例 程 24 ”代码 位 置 光盘 \TMVWO3\oavadminVbmglvadd depart_chkphp 


<? 
session start(); /开启 session 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../conn/conn.php": // 包 含 数据 库 链 接 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 


睛 “判断 输入 部 门 名 称 是 否 重复 */ 

ifisbool($_ POST[d_ name])){ 
echo "<script>alert( 名 称 已 存在 ， 请 重新 输入 !):history.go(-D:</script>": 
exit(); 


} 
必 确定 上 级 部 门 和 根部 门 */ 
这 $ POSTru idl != "0"){ 
$sqlstr = "select top_depart from tb_depart where id = ".$_POST[u_ id]: 
Sresult =Imysql query($sqlstr.$conn): 
Srows = mysql_fetch_row($result): 
if ($rows[top_depart] := 0) 
S$top_depart = Srows[top_depart]: 
else 
S$top_depart = $_POST[u id]: 
} 
else 
S$top_depart = 0: 
请 添加 新 部 门 */ 
S$sqlstr = "insert into tb_depart 
values(",".$_POST[d_name].".".$top_depart.".".$ POST[u id].",".$ POST[remark].")": 
Sresult = mysql query($sqlstr.Sconn): 
上 输出 添加 结果 */ 
re_message(S$result."show_depart.php") 
?> 
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3.10.5 ”单元 测试 


本 模块 使 用 了 JavaScript 和 自 定义 函数 来 显示 部 门 结构 列表 , 在 进行 测试 时 发 现 , 当 部 门 被 删除 到 
只 有 一 个 根部 门 时 ， 就 会 弹出 错误 提示 窗口 ， 如 图 3.32 所 示 。 


行 

字符 : 

鱼 误 : “0penllel ”未 定义 
人 i 


是 否 继 续 在 该 页 面 上 运行 脚本 程序 7 
[EL se | 
图 3.32 出错 信息 

单 击 “ 是 ”按钮 ， 其 他 操作 一 切 正 常 。 经 检查 发 现 ， 在 当前 页 面 被 载 入 时 调用 了 js 脚本 ， 语 句 

如 下 : 

<body onLoad="ShowMe(imgl,OpenMel1)"> 


在 部 门 被 删 到 只 有 一 个 时 ， 变 量 OpenMel 就 不 再 存在 ， 调 用 函数 ShowMe() 时 就 会 报错 ， 将 
onLoad="ShowMe(img1,OpenMe1)" 删 掉 ， 错 误 不 再 出 现 。 


3.11 系统 管理 模块 设计 


系统 管理 模块 在 后 人 台 系 统 中 占据 着 非常 重要 的 位 置 , 后台 其 他 的 模块 都 是 针对 前 台 功 能 而 设计 的 。 
而 系统 管理 是 针对 系统 本 身 的 一 些 信 息 、 情 况 而 设计 的 ， 一 般 包括 数据 备份 、 管 理 员 信息 修改 等 ， 甚 
至 可 以 对 操作 系统 的 配置 文件 进行 操作 ， 可 以 说 是 管理 员 的 管理 员 ， 是 后 台 系 统 中 必 不 可 少 的 组 成 
部 分 。 


3.11.1 系统 管理 模块 概述 


系统 管理 模块 包括 的 功能 有 : 

回 ”系统 日 志 : 记录 了 前 、 后 台 用 户 登录 时 的 相关 信息 。 
回 ”数据 备份 : 可 以 对 整个 数据 库 进行 备份 、 恢 复 和 删除 。 
加 ”修改 密码 :修改 后 台 管理 员 登 录 密码 。 

系统 管理 框架 如 图 3.33 所 示 。 
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(ean ) 


一 Csaz ) 


(Get ) 
~ 


图 3.33 系统 管理 用 例 图 
3.11.2 ”系统 管理 模块 技术 分 析 


在 系统 管理 模块 中 ， 保 存 数据 的 方式 使 用 了 数据 库 和 文件 ， 关 于 数据 库 技术 已 经 介绍 很 多 了 ， 本 
节 主 要 讲解 一 下 文件 及 文件 夹 的 相关 操作 。 

对 文件 的 操作 主要 分 3 步 : 打开 文件 ， 读 取 / 写 入 文件 ， 关 闭 文件 。 

1. 打开 文件 

打开 文件 使 用 的 是 fppen0 函 数 ， 函 数 的 一 般 格 式 如 下 : 

int fopen (string filename,string mode ); 

参数 filename 是 要 打开 的 包含 路 径 的 文件 名 ， 可 以 是 相对 路 径 或 绝对 路 径 。 如 果 没有 任何 前 绷 ， 
则 表示 打开 的 是 本 地 文件 ; 参数 mode 是 打开 文件 的 方式 , 常用 的 有 a (追加 )、w (只 写 ) 和 +r (只 读 )， 
代码 如 下 : 

$f_open = fopen("../bak/remark.sql"."at+"): 

2. 读 取 / 写 入 文件 

读 取 文件 的 函数 有 很 多 ， 最 常用 的 就 是 按 行 读 取 〈fgets0 函 数 ) 和 全 部 读 取 (readfile0 函 数 和 
fpassthru0 函 数 ) 。 

fgets() 函 数 的 一 般 格 式 为 : 

string feets ( int filename [ ,int length ]) : 

参数 filename 是 被 打开 的 文件 ， 参 数 length 是 要 读 取 的 数据 长 度 。 函 数 能 够 实现 从 filename 指定 
的 文件 中 读 取 一 行 并 返回 长 度 最 大 值 为 length-1 个 字 节 的 字符 串 。 在 遇 到 换行 符 、EOF 或 者 读 取 了 
length-1 个 字 节 后 停止 。 如 果 没 有 指定 length 的 长 度 ， 默 认 值 是 1KB。 代 码 如 下 : 


<?php 
S$f_open = fopen ("../bak/remak txt"."r") : /打开 文本 
echo fgets ( $f_open) : // 读 取 其 中 的 数据 


1 
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fclose ( $f open ) : 
2 


readfile() 函 数 的 一 般 格 式 如 下 : 
int readfile ($filename ): 
参数 $filname 为 要 打开 的 文件 名 ， 使 用 该 函数 时 ， 无 须 进 行 第 1、 第 3 步 ， 也 无 须 输出 语句 echo， 


直接 就 可 以 读 取 文件 的 全 部 内 容 并 显示 在 页 面 中 。 


3. 关闭 文件 
完成 对 文件 的 操作 后 ， 就 可 以 关闭 文件 了 。 关 闭 文件 的 命令 十 分 简单 ， 格 式 如 下 : 
fclose ( $open); 


其 中 参数 gopen 是 要 关闭 的 文件 资源 。 
在 本 模块 中 ， 还 用 到 了 两 个 自 定义 函数 c_ log0 函 数 和 show_file0 函 数 。c_log0 函 数 的 作用 是 删除 


日 志文 件 。 程 序 代码 如 下 : 


例 程 25 ”代码 位 置 : 光盘 \TM\03\oa\admin\inc\func.php 
function ¢ logO{ 


Sfilename="../log.txt"; // 日 志文 件 路 径 

if(file_ exists($filename)) // 判 断 文件 是 否 存 在 
unlink(Sfilename): /如 果 文 件 存在 ， 删 除 

else // 如 果 文 件 不 存在 ， 返 回 上 层 操作 


echo "<script>alert( 暂 无 系统 日 志 ! '"):history.go(-1):</script>"; 
} 


show_file0) 函 数 的 作用 是 返回 一 个 目录 下 的 所 有 文件 名 列表 , 并 以 数组 的 形式 返回 ,程序 代 码 如 下 : 


例 程 26 ”代码 位 置 ; 光盘 \TMVWO3\oavadminincfunc.php 
function show_fileO{ 


S$folder name ="../bak":; /数据 备份 目录 

S$d_open = opendir($folder name): /打开 备份 目录 

Snum = 0: /数组 下 标 

while(Sfile = readdir($d_open)){ // 循 环 读 取 文件 
Sfilename[$num] = $file: // 将 文件 名 存 为 数组 
Snumtt; 

} 

closedir($d_open): // 关 闭 目 录 

retum $filename; // 返 回 文件 数组 


} 


3.11.3 ”系统 日 志 的 实现 过 程 


通过 选择 导航 栏 中 的 “系统 管理 ”/“ 系 统 日 志 ” 菜 单项 可 进入 系统 日 志 页 面 。 在 该 页 面 中 显示 系 
统 日 志 信息 ， 在 系统 日 志 中 ， 记 录 着 前 后 人 台 , 包括 普通 用 户 和 后 台 管 理 员 所 有 人 员 登 录 系统 时 的 信息 ， 
包括 登录 时 间 、 用 户 账号 、 登 录 卫 等 。 这 些 信息 是 在 用 户 登录 时 系统 自动 添加 到 数据 表 中 的 (关于 用 
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户 登录 的 实现 请 参见 本 书 附带 的 光盘 ) 。 查 看 日 志 页 面 的 运行 结果 如 图 3.34 所 示 。 


蛙 录 陈 号 | 扯 录 时 间 Er 


一 一 2 下 poor-a-a 08:18:02| 127.0.0.1 | login 


pk poor-11-30 13:20:19h9e 168.1.237| locin 


= Toft poor-11-30 15:43:39| 127.0.0.1 | lc 
2 Taeg poor-11-30 16:14:49| 10001 | login | 


[A 


Bk pooT-11-30 16:39:48] 121.0.0.1 | login 


Bk oor-11-30 17:32:2390. 168.1.237| locin 


tsoft pooT-11-30 1T:34:0 127.0.0.1 | login 


3.34 系统 日 志 的 运行 结果 
显示 日 志 是 对 文件 进行 读 取 ， 定 位 到 日 志文 件 log.txt 后 ,使 用 fegs0 函 数 读 取 日 志 内 容 ， 并 显示 在 
页 面 中 。 程 序 代码 如 下 : 
例 程 27 ”代码 位 置 : 光盘 \TM\03\oa\admin\xtgl\slog.php 


<2php 
session_ start(); // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 

?> 

<Script src="../js/admin js.js"></script> /引入 js 脚本 文件 

<link href="../css/style.css" rel="stylesheet"> // 引 入 外 部 css 样 式 文件 
<t> 


<td colspan="4" height="30" align="center” valign="middle"><a href="del slog_chk.php” onclick="return 
del_chk0:"> 清 除 日 志 </a></td> 


</t> 
<?php 
Sfilename = "../log.txt": /日 志文 件 路 径 
if($f_ open = fopen($filename."r")) // 是 否 以 只 读 形式 打开 文件 
{ 
while($str = fgets($f_open,255)){ // 按 行 读 取 日 志文 件 内 容 
Schr = split(",".$str): // 以 “，” 为 分 隔 符 ， 返 回 数组 形式 的 记录 集 
echo "<tr>"; 
for($i = 0; $i < count($chr): SiHH){ /循环 输出 字段 值 
echo "<td align='center height="25'>".$chr[$i]."</td>": } 
echo "</tr>";} 
felose($f open): /关闭 文件 
} else // 如 果 没 有 日 志文 件 时 
echo "<script>alert( 还 没有 日 志文 件 ! '):history.go(-1):</script>"; 
?> 
</table> 
</td></tr></table> 
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对 于 无 用 的 日 志 信息 需要 定期 消除 ， 通 过 页 面 中 的 “清除 日 志 ” 超 链接 清空 日 志 ， 实 际 上 就 是 删 
除 日 志文 件 。 程 序 代码 如 下 : 
例 程 28 ”代码 位 置 : 光盘 \TMVWO3\oavadminxtglvdel slog chk .php 


<2php 
session start(): // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
c_log0; // 调 用 自 定义 函数 
echo "<script>alert(' 删 除 成 功 ! ):location='data_stock php':</script>"; 

?> 


3.11.4 数据 备份 的 实现 过 程 


在 病毒 肆虐 的 今天 ， 数 据 备份 的 重要 性 是 毋庸 置疑 的 。 没 有 任何 一 家 网 站 、 一 个 系统 敢 说 自己 是 
绝对 安全 的 ， 只 要 稍 不 留神 就 会 中 招 。 本 着 对 读者 负责 的 态度 ， 在 3.13 小 节 的 技术 专题 中 ， 将 详细 地 
介绍 MySQL 的 备份 与 恢复 ， 以 及 如 何在 php 中 对 MySQL 进行 备份 操作 。 这 里 先 简单 了 解 一 下 本 系统 
数据 备份 的 实现 过 程 。 数 据 备份 页 面 的 运行 结果 如 图 3.35 所 示 。 


办 公 自动 化 ee 


车 4 旧 备 从 
EE [eol. sal 
区 三 古 ] 「20071201. sal 可 


机 和 


3.35 数据 备份 的 运行 结果 


当 单 击 “ 备 份 数据 ”按钮 时 ， 系 统 执行 exec0 函 数 ， 该 函数 的 作用 是 执行 系统 命令 ， 通 过 系统 命令 
来 备份 数据 。 数 据 备份 的 程序 代码 如 下 : 


例 程 29 ”代码 位 置 : 光盘 \TMVWO3\oavadminxtglvdata_stock chkphp 


<2php 
session start(): // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 


Smysqlstr = "D:\\AppServ\IMySQL\\bin\mysqldump -uroot -h192.168.1.149 -proot --opt -B db_office 
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>../bak/".$_POST[b_namel: /| 数据 库 备 份 语句 
exec($mysqlstr): // 执 行 系统 命令 
echo "<script>alert(' 备 份 成 功 "):location='data_stock.php'</script>"; 
?> 


恢复 数据 和 备份 数据 的 执行 过 程 大 同 小 异 , 都 需要 使 用 exec0 函 数 执行 系统 命令 , 只 是 命令 执行 的 
操作 是 完全 相反 的 。 恢 复数 据 的 程序 代码 如 下 : 

例 程 30 ”代码 位 置 : 光盘 \TMWO3\oavadmin\xtglwvebak stock chkphp 

<?php 


session _start(): /开启 session 支 持 

include "../inc/chec.php": // 包 含 权限 检查 文件 

$mysqlstr = "D:\AppServ\IMySQL\\bin\mysql -uroot -h192.168.1.149 -proot db_office < ../bak/".$_ POST[r namel; 
/数据 恢复 命令 

exec($mysqlstr): // 执 行 系统 命令 


echo "<script>alert(' 恢 复 成 功 "):location='data_stock.php'</script>"; 
?> 


删除 数据 备份 实质 就 是 删除 备份 文件 ， 先 使 用 自 定 义 函数 show_file0 返 回 一 个 文件 名 列表 数组 ， 
再 使 用 for 循环 语句 ， 逐 一 删除 备份 文件 。 程 序 代 码 如 下 : 
例 程 31 代码 位 置 : 光盘 \TM\03\oa\admin\xtgl\del_stock_chk.php 


<?php 
session_start(); // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../inc/func.php": // 包 含 自 定义 函数 文件 
Sfilename = show_file(): // 调 用 自 定义 函数 show_file0， 返 回 文件 名 列表 


for($num = 2;$num < count($filename):Snum+H){ /循环 删除 文件 
unlink("../bak/".$filename[Snum]): 


echo "<script>alert(' 删 除 成 功 ! ):location='data_stock.php'</script>": 


3.12 开发 技巧 与 难点 分 析 


3.12.1 使 用 JavaScript 关联 多 选 列表 框 


在 本 系统 中 ， 多 个 模块 都 使 用 了 这 种 技术 ， 即 在 两 个 多 选 列表 框 之 间 产 生 关 联 ， 从 列表 框 A 中 删 
除 的 数据 被 相应 地 添加 到 列表 框 B 中 ， 反 之 从 列表 框 B 中 删除 的 数据 会 被 添加 到 列表 框 A 中 。 

在 权限 管理 模块 中 的 用 户 组 设置 ， 就 是 用 这 种 技术 实现 的 “添加 新 用 户 组 ”功能 ， 首 先 在 左 侧 的 
多 选 列表 框 中 选取 要 添加 的 成 员 ， 单 击 “ 添 加 组 员 ” 超 链接 ， 被 选中 的 成 员 就 被 添加 到 右 侧 列表 框 ， 
同时 删除 在 左 侧 列表 框 中 的 成 员 ， 运 行 结果 如 图 3.36 所 示 。 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


用 户 组 : 隔 可 组 用 户 组 : 隔 可 组 
本 eee 
aaa 
尖 加 组 员 >》 天 加 组 员 >》 
《< 出 除 组 员 《< 各 除 组 员 
EE 


图 3.36 使 用 JavaScript 关 联 多 选 列表 框 


下 面 来 看 一 下 具体 的 操作 。 首 先 创建 一 个 表单 ， 其 中 两 个 多 选 列表 框 为 必 选 的 表单 元 素 ， 分 别 命 
名 为 left 和 right， 放 在 左 、 右 两 边 ， 其 中 left 列表 框 中 的 数据 是 从 数据 库 中 读 取 的 ，right 列表 框 则 初 
始 为 室 。 程 序 代码 如 下 : 


例 程 32 ”代码 位 置 : 光盘 \TMNWO3\oavadminWqxglvadd_group.php 


<!-- 输 出 左 侧 列表 框 --> 
<SELECT name="left" size="10" multiple style= "width:100px: "> 
<2php 
while($rows = Imysql_fetch row(S$resulD){ 
echo “<option value=". $rows[1].">".$rows[1]."</option>"; // 从 数据 库 中 读 取 列表 值 


} 
?> 
</SELECT> 
A 
</td> 
<!--“ 添 加 ”、“ 删 除 ” 超 链接 -> 
<td width="96" align="center" valign="middle"> 
<a href="#" onClick="activeList(document.form1.left,document.form1.right)"> 添 加 组 员 &gt;&gt;</a><br> 
<br> 
<a href="#" onClick="activeList(document.form1.right,document.form1.leff)">&lt:&lt: 删 除 组 员 </a></td> 
<!-- 右 侧 列表 框 --> 
<td colspan="2" align="center" valign="middle"><select name="right" size="10" multiple style="width:100px: "> 
</select></td> 
</t> 
<h> 
<td height="30" colspan="4" align="center" valign="middle"> 
<input type="hidden" name="g_list" /> 
<L- “添加 ”按钮 -> 
<input type="submit" value=" 添 加 " onclick="return glist0" /><input type="reset" value=" 重 置 " /> 


上 面 的 代码 主要 使 用 的 是 HTML 和 PHP 语言 ， 和 普通 的 页 面 没 有 什么 区 别 , 核心 的 语句 就 是 代码 
中 的 加 粗 部 分 〈 不 是 注释 ) ， 当 单 击 “ 添 加 组 员 ” 或 “删除 组 员 ” 超 链接 时 ， 将 触发 JavaScript 脚本 事 
件 ， 执 行 activeListO 函 数 。 函 数 的 代码 如 下 : 

例 程 33 ”代码 位 置 : 光盘 \TM\03\oa\admin\js\admin js.js 


/#* 关 联 多 选 列 表 
*headStream: 源 多 选 列 表 
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*endStream: 目的 多 选 列 表 
SF 
function activeList(headStream.endStream){ 
var valueList = new Array(): 
var textList = new Array0: 
var value TmpText = new Array():; 
var index = 0; 
让 ”存储 源 列表 中 的 被 选中 的 数据 */ 


for(var i=0; i<headStream.options.length: i++){ 
if(headStream.options[i].selected){ 


valueList[index] = headStream.options[i].value; 
textList[index] = headStream.options[i] .text: 
valueTmpText[valueList[index]] = headStream.options[j]: 


index ++; 


局 
谨 ” 向 目的 列表 中 添加 数据 ， 同 时 删除 对 应 的 源 列表 数据 


for(var i=0; i<textListlength: i++) { 


Var foption = document.createElement("option'"): 


foption.text = textList[i]: 
foption.value = valueList[i]: 
endStream.add(foption): 


headStream.removeChild(valueTmpText[valueList[i]]): 


和 


3.12.2 用户 组 设置 


在 后 台 的 权限 管理 模块 中 ， 放 弃 了 常用 的 “ 读 ”、 


// 数 组 ， 用 来 存储 要 移动 数据 的 value 值 
// 数 组 ， 用 来 存储 要 移动 数据 的 text 值 
/数组 ， 用 来 存储 要 移动 数据 的 option 项 
/数组 下 标 


// 判 断 元 素 是 否 被 选中 
// 存 储 value 值 

// 存 储 text 值 

// 存 储 option 项 


// 建 立新 的 option 项 


// 向 目标 列表 添加 新 建 option 项 
// 从 源 列 表 中 移 除 option 项 


“ 写 ” 这 样 的 常规 操作 ， 而 是 采用 了 用 户 组 的 


模式 。 用 户 组 的 设置 主要 是 两 方面 : 一 方面 是 用 户 组 成 员 ， 同 一 用 户 可 以 分 配 在 不 同 的 用 户 组 ， 另 一 
方面 是 功能 列表 ， 可 以 设置 用 户 组 的 访问 权限 ， 不 同 的 用 户 组 ， 所 看 到 的 功能 列表 也 会 不 同 。 这 样 ， 
可 以 使 用 户 的 权利 更 加 细 化 , 防止 居心 区 测 的 用 户 恶 意 破坏 。 下 面 是 用 户 组 A 和 用 户 组 B 的 对 比 结果 ， 


如 图 3.37 所 示 。 


| 功能 分 关 号 1 


骸 区 


| ， 功能 分 类 有 展区 


图 3.37 用 户 组 权限 对 比 
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在 功能 列表 的 设置 中 ， 首 先 默认 所 有 的 功能 都 是 可 访问 的 ， 当 有 的 功能 需要 进行 限制 时 ， 就 将 无 
用 的 功能 从 用 户 组 中 删 掉 。 程 序 代码 如 下 : 
例 程 34 ”代码 位 置 : 光盘 \TM\03\oa\admin\qxgl\pur assign.php 


<?php 
session start(): // 开 启 session 支 持 
include "../inc/chec.php": // 包 含 权限 检查 文件 
include "../conn/conn.php": // 包 含 数据 库 链 接 文 件 
$sqlstr = "select id,f name,o group from tb list"; // 生 成 SQL 语 句 

?> 

<script sre="../js/admin js.js"></script> // 引 入 js 脚本 文件 


<link href="../css/style.css" rel="stylesheet"> // 引 入 外 部 css 样 式 文件 
<tr><td width="75" height="20" align="center" valign="middle"> 功 能 </td> 
<td width="75" height="20" align="center” valign="middle"> 开 放 组 </td> 
<td width="75" height="20" align="center”" valign="middle"> 操 作 </td></t> 


<?php 
入 ”输出 功能 列表 项 */ 
Sresult = mysql_query($sqlstr,$conn); 
while($rows = mysql fetch row($result)){ 
echo "<tr>"; 
for($i= 1:$i < count($rows): $i+H+){ 
echo "<td align=center valign=middle>".$rows[$i]."</td>"; 


} 
echo "<td align=center valign=middle><a href='modify_assign.php?id=".$Srows[0]."> 修 改 </a></td>"; 
echo "</tr>"; 


/ 兴 认 突 闪光 实 闪 光 兴 突 突 交 光山 实 次 闪闪 闪闪 闪闪 实 次 内/ 
?> 

</table> 

</td></tr></table> 


本 节 是 3.12.1 节 关 联 多 选 列表 的 延伸 ， 是 具体 应 用 关联 列表 实现 的 一 项 功能 。 在 实际 的 应 用 中 ， 
请 多 多 开动 脑筋 ， 才 能 用 平凡 的 技术 实现 不 平凡 的 功能 。 


3.13 ”MySQL 数据 备份 专题 


关于 备份 数据 的 重要 性 ， 这 里 就 不 再 多 说 了 。 本 节 先 介绍 常用 数据 库 备份 的 方式 ， 再 讲解 一 下 
MySQL 数据 备份 与 恢复 的 方式 方法 ， 最 后 看 一 看 在 PHP 中 是 如 何 对 MySQL 数据 进行 备份 的 。 

1. 数据 库 备 份 类 型 

(1) 完全 备份 

这 是 大 多 数 人 常用 的 方式 ， 它 可 以 备份 整个 数据 库 ， 但 需要 花费 更 多 的 时 间 和 空间 ， 所 以 一 般 推 
荐 一 周 做 一 次 。 

(2) 事务 日 志 备份 
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事务 日 志 就 是 对 数据 库 操作 的 记录 ， 它 记录 数据 库 的 改变 ， 备 份 的 时 候 只 需要 备份 自 上 次 备份 以 
来 对 数据 库 所 做 的 改变 ， 所 以 只 需要 很 少 的 时 间 。 推 荐 每 小 时 一 次 。 

(3) 差异 备份 

也 叫 增 量 备份 。 它 是 另 一 种 只 备份 一 小 部 分 数据 库 的 方法 ， 它 不 使 用 事务 日 志 ， 相 反 ， 它 使 用 整 
个 数据 库 的 一 种 新 映 象 。 它 比 最 初 的 完全 备份 小 ， 因 为 它 只 包含 自 上 次 完全 备份 以 来 所 改变 的 数据 库 。 
它 的 优点 是 存储 和 恢复 速度 快 。 推 荐 每 天 做 一 次 。 

(4) 文件 备份 

数据 库 可 以 由 硬盘 上 的 许多 文件 构成 。 如 果 这 个 数据 库 非常 大 ， 恐 怕 一 个 晚上 也 不 能 将 它 备 份 完 ， 
可 以 每 天 备份 数据 库 的 一 部 分 。 由 于 一 般 情况 下 数据 库 不 会 大 到 必须 使 用 多 个 文件 存储 ， 所 以 这 种 备 
份 不 是 很 常用 。 

2. MySQL 数据 备份 与 恢复 

MySQL 中 的 备份 是 非常 灵活 的 , 既 可 以 对 单独 的 数据 表 进 行 备份 , 也 可 以 对 整个 数据 库 进行 备份 ， 
下 面 分 别 进行 介绍 。 

(1) 数据 表 备 份 

备份 数据 表 的 语法 如 下 : 

SELECT* INTO OUTFILE 'file name' FROM tbl name 

回 file name: 备份 的 文件 名 。 如 user.sql。 

回 tbl_name: 备份 的 数据 表 名 。 如 tb_users。 

使 用 该 备份 语句 首先 要 登录 到 MySQL， 并 进入 到 相对 应 的 数据 库 中 ， 如 要 备份 的 表 tb_users 是 属 
于 数据 库 db_office 的 ， 备 份 文件 名 为 users.sql， 则 完整 的 备份 流程 如 图 3.38 所 示 。 
| 口 | 妆 | 


图 3.38 数据 表 备份 


备份 后 的 文件 保存 在 数据 表 所 在 的 目录 中 , 如 本 例 中 表 tb_users 所 在 的 目录 为 D:\AppSerwWMySQIA 
datavdb_office， 则 备份 的 sql 文件 就 保存 在 这 个 路 径 下 。 
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如 果 要 恢复 备份 文件 ， 代 码 如 下 : 
LOAD DATA INFILE ser.sql REPLACE into table 'tb_users' 
(2) 数据 库 备份 

数据 库 备份 时 ， 可 以 不 用 登录 到 sql 里 ， 直 接 使 用 mysqldump 命令 即 可 。 该 命令 的 格式 如 下 : 

mysqldump -uroot -proot db_office > c:\\remark.txt 

备份 的 数据 表 ， 实 际 就 是 一 个 完整 的 建 库 流程 ， 如 果 要 恢复 备份 ， 直 接 使 用 MySQL 命令 即 可 ， 
该 语句 如 下 : 

mysql -uroot -proot db_office < c:\\remark.txt 

mysqldump 语句 的 功能 十 分 强大 ， 备 份 只 是 其 中 的 一 个 功能 ， 感 兴趣 的 读者 可 以 参考 mysqldump 
的 帮助 文档 。 

3. 在 PHP 中 备份 数据 库 


在 PHP 中 备份 数据 库 ， 主 要 通过 exec() 函 数 执行 系统 命令 。 
exec0 函 数 的 格式 为 : 


string exec ( string command ) 
参数 command 为 要 执行 的 系统 命令 。 这 里 为 备份 数据 表 的 命令 ， 代 码 如 下 : 
exec("D:\AppServ\IMySQL\\bin\mysqldump -uroot -h192.168.1.149 -proot --opt -B db_office > ../bak/remark.sql"): 


名 注意 : 虽然 在 安装 时 MySQL 已 经 设置 了 环境 变量 ， 但 如 果 要 在 PHP 中 执行 mysqldump、mysql 
命令 ， 则 必须 给 出 命令 的 完整 路 径 ， 否 则 ， 就 会 出 现 文件 大 小 为 0KB 的 结果 。 


3.14 本 章 总 结 


本 章 通 过 一 个 完整 的 办 公 自 动 化 管理 系统 开发 过 程 ， 向 读者 讲述 了 系统 开发 的 儿 个 重要 环节 ， 还 
介绍 了 几 个 主要 模块 的 功能 实现 ， 最 后 又 着 重 讲解 了 MySQL 备份 的 相关 知识 。 相 信 通 过 本 章 的 学 习 ， 
读者 对 如 何 开 发 一 个 真实 的 项 目 有 了 进一步 的 了 解 。 相 信 读 者 通过 自己 动手 实践 ， 完 成 一 个 功能 相似 
的 办 公 自 动 化 管理 系统 是 完全 不 成 问题 的 。 


第 章 


电子 商务 平 合 网 


(Apache+PHP+phpMyAdmin+MYySQL 5.0 实现 ) 
( 铝 " 视频 讲解 : 1 小 时 26 分钟 ) 


电子 商务 源 于 英文 ELECTRONIC COMMERCE， 简 写 为 EC。 顾名思义 ， 其 内 
容 包含 两 个 方面 : 一 是 电子 方式 ， 二 是 商贸 活动 。 

EC (电子 商务 ) 指 的 是 利用 简单 、 快 捷 、 低 成 本 的 电子 通信 方式 ， 买 卖 双 方 不 
谋面 而 在 网 上 开展 的 各 种 商贸 活动 。 在 全 球 知 识 经 济 和 信息 化 高 速 发 展 的 今天 ， 信 
息 化 是 决定 企业 成 败 的 关键 因素 , 信息 的 有 效 利用 成 为 新 经 济 模式 中 企业 增强 竞争 
力 的 重要 手段 。 

电子 商务 作为 一 种 崭新 的 商务 运作 模式 ， 越 来 越 受 到 企业 的 重视 。 电 子 商务 的 
魅力 在 于 它 能 打 碎 现存 的 一 切 链 条 结构 ， 让 产品 群 、 客 户 群 、 技 术 群 、 物 流 群 等 重 
新 排队 、 优 化 组 合 ， 为 企业 业务 的 重新 组 合 提供 无 限 商机 ， 开 辟 新 的 竞争 领域 ， 形 
成 新 的 利益 分 配 格局 。 本 章 通 过 开发 一 个 流行 的 电子 商务 网 站 一 一 电子 商务 平台 
网 ， 快 速 开 发 一 个 电子 商务 平台 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


WI 项 目 需求 分 析 和 系统 设计 的 方法 MH” 打印 订单 技术 
WI 在 MySQL 中 创建 数据 库 、 数 据 表 的 不 同方 法 MH 单元 测试 的 方法 
MW 如何 设计 公共 类 MW 网 站 安全 技术 
”实现 购物 车 的 原理 MH ”加 密 技术 
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4.1 开发 背景 


随 着 我 国 网 络 经 济 的 快速 发 展 ， 互 联网 用 户 数 逐 日 增多 ， 有 过 网 络 购物 经 历 的 用 户 达 到 3000 万 人 
以 上 ， 其 中 有 一 半 人 已 经 习惯 网 上 购物 ， 而 且 这 个 数目 正在 快速 增长 。 以 商品 销售 为 主 的 某 商 城 目前 
正面 临 竞 争 和 效益 下 降 的 压力 , 每 天 的 内 部 工作 流程 都 需要 花费 大 量 成 本 ,为 了 不 受 传统 方式 的 约束 ， 
减少 过 多 成 本 和 人 员 的 开销 ， 增 强 企业 的 竞争 力 ， 该 商城 决定 采用 电子 商务 模式 ， 向 多 元 化 发 展 ， 借 
助 Internet 在 国内 的 快速 发 展 ， 在 建立 企业 宣传 网 络 的 同时 ， 也 逐步 扩大 企业 自身 的 网 络 销售 渠道 ， 聚 
集 部 分 资金 投入 网 站 建设 。 通 过 网 上 交易 为 主要 形式 ， 进 行 网 络 交易 的 过 渡 ， 带 动 商城 的 快速 发 展 ， 
快速 提高 企业 的 经 济 效益 。 企 业 通过 建立 自己 独 有 的 网 上 交易 平台 ， 为 消费 者 提供 安全 、 便 捷 的 购物 
方式 ， 为 商家 提供 交易 处 理 和 丰富 的 管理 功能 。 现 需要 委托 某 单位 开发 一 套 完整 的 电子 商务 管理 平台 
从 而 能 以 低 成 本 为 消费 者 提供 更 快 更 好 的 服务 。 


4.2 系统 分 析 


在 开发 一 个 项 目 之 前 ， 首 先 要 对 所 开发 的 项 目 进 行 需求 分 析 、 可 行 性 分 析 ， 以 使 项 目 开 发 人 员 了 
解 和 掌握 网 站 的 前 期 策划 和 网 站 开发 流程 。 


4.2.1 需求 分 析 


随 着 Internet 的 发 展 ， 电 子 商 务 将 成 为 21 世纪 网 络 发 展 的 主流 ， 网 上 购物 将 成 为 一 种 购物 时 尚 ， 
它 为 人 们 提供 了 网 络 购物 的 方便 性 ， 使 顾客 足 不 出 户 就 可 以 购买 商品 。 现 在 流行 的 电子 商务 有 B2B、 
B2C、C2C、G2C 等 类 型 。 电 子 商务 平台 网 是 建立 在 企业 与 消费 者 (B2B 类 型 ) 之 间 的 商务 交易 网 站 ， 
它 可 以 使 顾客 通过 使 用 浏览 商品 、 网 络 购物 、 查 询 订单 、 打 印 订单 和 查看 公告 等 功能 购买 自己 所 需 的 
商品 。 通 过 对 一 些 典 型 电子 商城 网 站 的 考察 、 分 析 ， 并 结合 企业 要 求 以 及 实际 的 市 场 调查 ， 要 求 本 系 
统 具 有 以 下 功能 : 

网 站 设计 页 面 要 求 美观 大 方 、 个 性 化 ， 能 够 展示 企业 形象 。 

网 站 页 面具 有 banner 广告 ， 树 立 企业 良好 的 口碑 宣传 。 

设计 完全 从 网 络 营销 的 角度 设计 ， 不 花 钱 就 可 以 被 世界 各 大 搜索 引擎 收录 。 
企业 所 有 的 商品 数据 都 在 电子 商务 平台 中 展示 。 

规范 、 完 善 的 基础 信息 设置 。 

商品 分 类 详尽 ， 可 按 不 同类 别 查 看 商品 信息 。 

按 商 品 大 类 及 商品 名 称 、 订 单 进行 模糊 查询 。 

实现 选 购 商品 一 订购 商品 一 收银 结账 一 打印 订单 功能 。 

实现 各 种 查询 ， 如 模糊 查询 、 高 级 查询 等 。 


办 办 办 办 办 凶 多 凶 罗 
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回 ”管理 员 对 用 户 订单 进行 管理 。 
回 ”提供 了 交互 式 的 销售 渠道 。 使 商家 能 及 时 得 到 顾客 反馈 信息 ， 了 解 顾 客 需求 ， 改 进 本 身 的 工作 。 


4.2.2 ”可行 性 分 析 


可 行 性 分 析 的 目的 就 是 要 用 最 小 的 代价 在 尽 可 能 短 的 时 间 内 确定 问题 是 否 能 够 解决 ， 当 然 不 能 靠 
主观 猜想 而 是 要 靠 客观 分 析 。 必 须 分 析 解 法 的 利弊 ， 从 而 判定 系统 目标 和 规模 是 否 现实 ， 系 统 完成 后 
所 能 带 来 的 效益 是 否 大 到 值得 去 投资 开发 这 个 系统 的 程度 。 因 此 ， 可 行 性 分 析 实 质 上 是 要 进行 依次 大 
大 地 压缩 简化 了 的 系统 分 析 和 设计 的 过 程 ， 也 就 是 在 较 高 层次 以 较 抽象 的 方式 进行 的 系统 分 析 和 设计 
的 过 程 。 

该 软件 项 目 可 行 性 研究 报告 是 对 项 目 课题 的 全 面 通 盘 考 虑 ， 是 项 目 分 析 员 进 一 步 工 作 的 前 提 ， 是 软件 
开发 人 员 正 确 成 功 地 开发 项 目的 前 提 与 基础 。 编 写 可 行 性 研究 报告 可 以 使 软件 开发 团体 尽 可 能 早 地 估计 研 
制 课题 的 可 行 性 ， 可 以 在 定义 阶段 较 早 地 认识 到 系统 方案 的 缺陷 ， 从 而 节约 开发 时 间 和 精力 、 节 省 资金 ， 
并 且 避 免 了 许多 专业 方面 的 困难 ， 所 以 特意 编写 该 软件 项 目 可 行 性 研究 报告 ， 意 在 起 到 事半功倍 的 效果 。 

电子 商务 平台 网 的 可 行 性 可 从 以 下 两 点 考虑 : 
回 ”经济 可 行 性 分 析 。 电 子 商务 网 站 的 宗 则 是 根据 用 户 需 求 和 市 场 形势 ， 提 供 商 品 的 详细 信息 ， 
并 对 商品 进行 详细 分 类 ， 方 便 用 户 查找 和 购买 所 需 的 商品 。 

回 ”技术 可 行 性 分 析 。 电 子 商 务 网 站 提供 购物 车 和 收银 台 功 能 ， 用 户 选择 商品 并 可 以 在 线 提 交 订 
单 ; 信息 管理 系统 实现 对 商品 信息 、 用 户 信息 、 订 单 信息 以 及 交易 制度 等 的 管理 ， 使 网 站 具 
有 友好 的 交易 界面 和 良好 的 管理 平台 。 


43 系统 设计 


4.3.1 系统 目标 


目前 Internet 网 上 商家 不 少 , 但 由 于 缺乏 相应 的 安全 保障 、 管 理 机 制 、 便 捷 的 操作 和 可 维护 性 ， 造 
成 重复 建设 和 资源 浪费 。 一 个 网 上 购物 网 站 ， 尤 其 是 数据 流量 比较 大 的 网 络 管理 系统 ， 必 须要 满足 使 
用 方便 、 操 作 灵 活 等 设计 需求 。 根 据 需 求 分 析 的 描述 以 及 与 用 户 的 沟通 ， 本 系统 在 设计 时 应 该 满足 以 
下 几 个 目标 ; 
系统 采用 人 机 对 话 方式 ， 界 面 设计 美观 大 方 、 方 便 、 快 捷 、 准 确 ， 数 据 存储 安全 可 靠 。 
全 面 展示 商城 内 所 有 商品 ， 并 可 以 展示 最 新 商品 、 推 荐 商品 、 热 门 商品 。 
实现 各 种 查询 ， 如 模糊 查询 、 高 级 查询 等 。 
查看 商城 内 的 公告 信息 。 
灵活 快速 的 填写 供求 信息 ， 使 信息 传递 更 快捷 。 
为 充分 展现 网 站 的 交互 性 ， 本 系统 实现 “网 上 用 户 订购 一 支付 一 发 货 ” 一 条 链 路 。 
实现 订单 打印 功能 。 


办 办 办 办 办 
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对 用 户 输入 的 数据 ， 系 统 进行 严格 的 数据 检验 ， 尽 可 能 排除 人 为 的 错误 。 
支持 友情 链接 功能 。 

网 站 最 大 限度 地 实现 了 易 维 护 性 和 易 操作 性 。 

系统 运行 稳定 、 安 全 可 靠 。 


回回 加 加 


4.3.2 ”系统 功能 结构 


为 了 使 读者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 电子 商务 网 站 的 前 台 功 能 模块 结构 图 和 后 台 
功能 模块 结构 图 。 
电子 商务 平台 网 前 台 管理 系统 的 功能 设计 如 图 4.1 所 示 。 


电子 商务 平台 前 台 管理 系统 


图 4.1 前 台 功 能 模块 结构 图 
电子 商务 平台 网 后 台 管 理 系 统 的 功能 设计 如 图 4.2 所 示 。 


电子 商务 平台 后 台 管理 系统 


商品 管理 | |[ 用 户 管理 | | EE | [ 信息 管理 ] 


邢 吉 车 菏 
盈 融 中核 
芝 沪 对 疡 
[Er 


起 
上 吐 亢 刘 并 
崩 吐 叫 昭 了 型 
兽 刘 溯 峙 囊肿 间 
性 所 撞车 站 站 
沁 击 
赠品 哩 以 


月 奢 池 并 台 下 


图 4.2 ”后台 功能 模块 结构 图 
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4.3.3 ”购物 流程 图 


所 谓 电子 商务 ， 实 质 就 是 建立 一 个 虚拟 的 购物 超市 ， 当 在 超市 选 购 商 品 时 ， 首 先 应 将 商品 放 到 购 
物 车 中 ， 待 挑选 好 所 有 商品 之 后 就 可 以 到 收银 台 去 开 收 货 单 ， 根 据 收 货 单据 付款 。 制 作 电 子 商 务 系统 
的 原理 与 在 超市 购物 的 原理 是 一 样 的 。 首 先 客户 应 该 在 网 页 中 选 购 自 己 需要 的 商品 并 将 商品 放 入 购物 
车 中 ， 当 然 也 可 以 改变 购买 商品 的 数量 或 清空 购物 车 中 的 商品 。 选 购 好 商品 后 就 可 以 到 收银 台 进 行 填 
写 收 货 人 信息 、 提 交 收 货 人 信息 、 查 看 账单 等 操作 。 

由 于 系统 的 定位 是 一 个 网 上 的 购物 系统 ， 是 一 个 电子 商务 类 网 站 。 传 统 的 C/S〔 客 户 机 /服务 器 ) 
已 经 不 适应 了 ， 作 为 Internet 上 的 Web 应 用 ， 需 要 的 是 B/S (浏览 器 /服务 器 ) 架构。 根据 上 面 的 分 析 ， 
笔者 将 采用 PHP 这 种 相对 流行 而 且 安全 性 较 高 的 Web 开发 语言 ， 同 时 使 用 PHP 的 黄金 搭档 MySQL 
作为 后 台数 据 库 。 电 子 商 务 平台 的 购物 流程 如 图 4.3 所 示 。 


浏览 网 站 
进行 商品 信息 管理 、 其 他 功能 模块 
订单 管理 、 用 户 管理 、 


网 站 信息 管理 是 


图 4.3 电子 商务 平台 的 购物 流程 图 
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4.3.4 系统 预览 


电子 商务 平台 网 由 多 个 程序 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 可 参见 光盘 中 的 源 
程序 。 

前 台 首 页 如 图 4.4 所 示 ， 该 页 面 用 于 实现 商品 信息 展示 、 用 户 登 录 、 公 告 信息 、 友 情 链 接 、 商 品 信 
息 查 询 等 功能 。 后 台 首 页 如 图 4.5 所 示 ， 该 页 面 用 于 实现 查看 订单 、 执 行 订单 、 删 除 订单 、 打 印 订 单 等 
功能 。 


图 4.4 前 台 首 页 (光盘 \TM\04\shop\index.php) 图 4.5 后 台 首 页 (光盘 \TM\04\shop\admin\default.php) 
会 员 信息 管理 页 面 如 图 4.6 所 示 ， 该 页 面 用 于 实现 修改 个 人 信息 、 修 改 密码 、 用 户 留 言 、 用 户 注销 


等 功能 。 购 物 车 列表 页 面 如 图 4.7 所 示 ， 该 页 面 用 于 实现 查看 购物 车 、 移 除 购物 商品 、 修 改 商 品 数量 、 
清空 购物 车 、 收 银 台 结账 、 生 成 订单 等 功能 。 


了 mW 
SEE | OA | NA | TNA | MAN | NPP | TOR | Wo fp ER 


EE 


图 4.6 会 员 信息 管理 (光盘 \TM\04\shop\usercenter.php) 图 4.7 购物 车 列表 (光盘 \TM\04\shop\gouwul.php) 


用 户 订单 查询 页 面 如 图 4.8 所 示 , 该 页 面 用 于 实现 按 下 单 人 或 订单 号 进行 订单 信息 查询 功能 。 管理 
员 登 录 页 面 如 图 4.9 所 示 ， 该 页 面 用 于 实现 对 管理 员 登 录 的 用 户 名 和 密码 进行 验证 等 功能 。 
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图 4.8 ”用 户 订单 查询 (光盘 \TM\04\shop\admin\findddphp) 图 4.9 管理 员 登 录 (光盘 \TM\04\shop\admin\index.php) 


4.3.5 ”开发 环境 


在 开发 电子 商务 平台 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
.服务 器 端 

操作 系统 ，Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.2。 

PHP 软件 : PHP 5.1.6。 

数据 库 : MySQL 5.0.24。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.9.0.2。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效果 1024X768 像素 。 

客户 端 ; 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效果 1024X 768 像素 。 


4.3.6 ”文件 夹 组 织 结构 


二 


园 回回 回回 网 网 网 


加 加 


在 进行 网 站 开发 前 ， 首 先 要 规划 网 站 的 架构 。 也 就 是 说 ， 建 立 多 个 文件 夹 ， 对 各 个 功能 模块 进行 
划分 ， 实 现 统一 管理 ， 这 样 做 易于 网 站 的 开发 、 管 理 和 维护 。 本 案例 的 站 点 管理 规划 如 图 4.10 所 示 。 


用 于 存储 网 站 后 台 文件 

用 于 存储 网 站 前 台 使 用 的 CSS 样式 表 

用 于 存储 后 台数 据 库 连 接 文件 

用 于 存 鳍 网 站 后 台 页 面 使 用 的 图 片 文件 

用 于 存储 网 站 后 台 页 面 使 用 的 上 传 图 片 文件 
用 于 存 竺 网 站 前 台 使 用 的 CSS 样式 表 

用 于 存储 数据 库 文件 

用 于 存 鳍 网 站 前 台 页 面 使 用 的 图 片 文件 

用 于 存储 前 台数 据 库 连 接 文件 


图 4.10 文件 夹 组 织 结构 
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4.4 数据 库 设 计 


电子 商务 平台 网 是 一 个 数据 库 开 发 应 用 程序 。 本 节 针 对 电子 商务 平台 网 的 数据 库 设 计 进 行 详 细 
介绍 。 


4.4.1 数据 库 分析 


因为 本 项 目 面向 的 是 为 中 小 型 的 图 书馆 ， 所 以 需要 充分 考虑 到 成 本 及 用 户 需 求 〈 如 跨 平 台 ) 等 问 
题 ， 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 的 跨 平 台 的 关系 型 数据 库 系 统 ， 
这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 作 为 PHP 的 黄金 搭档 MySQL 数 
据 库 不 仅 存储 和 管理 功能 强大 ， 而 且 它 是 完全 免费 使 用 的 ， 很 多 网 站 都 可 以 下 载 到 它 ， 这 样 可 以 为 企 
业 节 省 很 大 一 部 分 开支 。PHP 中 也 提供 了 强大 的 支持 MySQL 数据 库 的 函数 ，phpMyAdmin 为 MySQL 
数据 库 提 供 了 图 形 化 界面 。 


4.4.2 ”数据库 概念 设计 


根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设 计 ， 规 划 出 电子 商务 平台 的 实体 关系 E-R 图 。 实 体 
关系 E-R 图 是 用 来 描述 实体 之 间 关 系 的 图 表 ， 构 成 E-R 图 的 基本 要 素 是 实体 型 、 属 性 和 联系 ， 其 表示 
方法 为 : 

实体 型 : 用 矩形 表示 ， 拢 形 框 内 标注 实体 名 。 

属性 : 用 椭圆 形 表示 ， 并 用 无 向 边 将 其 与 相应 的 实体 连接 起 来 。 

联系 : 用 菱形 表示 ， 攻 形 框 内 标注 联系 名 ， 并 用 无 向 边 分 别 与 有 关 实 体 连接 起 来 ， 同 时 在 无 

向 边 旁 标 上 联系 的 类 型 (1:1、1:n 或 m:n) 。 

本 系统 中 使 用 的 数据 库 实体 分 别 为 商品 信息 实体 、 商 品类 型 实体 、 用 户 信 息 实体 、 用 户 订 单 实 体 、 
用 户 留言 实体 、 商 品评 价 实体 、 管 理 员 信息 实体 、 公 告 信 息 实 体 和 友情 链接 实体 。 下 面 将 介绍 几 个 关 
键 实体 的 E-R 图 。 

1. 商品 信息 实体 

商品 信息 实体 包括 编号 、 名 称 、 价 格 、 上 市 时 间 、 等 级 、 型 号 、 图 片 路 径 、 数 量 、 购 买 次 数 、 是 
否 推荐 、 商 品类 型 、 会 员 价 、 市 场 价 、 商 品 品牌 等 属性 。 商 品 信息 实体 的 E-R 图 如 图 4.11 所 示 。 

2. 商品 订单 实体 

商品 订单 实体 包括 编号 、 订 单 号 、 商 品 串 、 数 量 串 、 收 货 人 姓名 、 收 件 人 性 别 、 送 货 地 址 、 邮 编 、 
联系 电话 、E-mail、 收 货 方式 、 支 付 方式 、 用 户 留言 、 下 单 时 间 、 下 单 人 姓名 、 订 单 状态 、 价 格 总 计 等 
属性 。 商 品 订单 实体 的 E-R 图 如 图 4.12 所 示 。 
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图 4.12 商品 订单 实体 E-R 图 
3. 用 户 信息 实体 


用 户 信息 实体 包括 编号 、 用 户 名 、 加 密 密 码 、 冻 结 标记 、E-mail、 身 份 证 号 、 联 系 电话 、QQ 号 、 
密码 提示 、 密 码 答 案 、 邮 编 、 注 册 时 间 、 真 实 姓 名 、 密 码 等 属性 。 用 户 信息 实体 的 E-R 图 如 图 4.13 
所 示 。 


图 4.13 用 户 信 息 实体 E-R 
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4. 商品 评价 实体 
商品 评价 实体 包括 编号 、 用 户 编号 、 商 品 编号 、 评 价 主题 、 评 价 内 容 、 评 价 时 间 等 属性 。 商 品评 
价 实体 的 E-R 图 如 图 4.14 所 示 。 


评价 时 间 


图 4.14 商品 评价 实体 E-R 
4.4.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需求 的 分 析 ， 电 子 商 务 系统 db_shop 数据 库 主 要 包含 如 下 9 个 数据 表 ， 如 
图 4.15 所 示 。 


加 服务 器 :localhost ， 轧 数据 库 : db_shop 

表 类 型 整理 说 明 
th_admin MYISAM = gb2312_chinese_ci 管理 员 信 息 表 
tb_dingdan 。 MYISAM 。 gb2312_chinese_cl 订单 信息 表 
th_gonggao MySAM gb2312_chinese_ci 公告 信息 表 
th_leaveword MSAM 。 gb2312_chinese_cl 用 户 留言 信息 表 
tb Jinks MYISAM = gb2312_chinese_ci 友情 链接 信息 表 
tb_pingjia MyISAM gb2312_chinese_cl 商品 评价 信息 表 
th_shangpin MySAM gb2312_chinese_cl 商品 信息 表 
th_type MYISAM 。 gb2312_chinese_ci 商品 关 型 信息 表 
th_user MyISAM gb2312_chinese_ci 用 户 信息 表 


4.15 电子 商务 系统 数据 表 


下 面 以 db_shop 数据 库 中 的 tb_admin 数据 表 为 例 ， 通 过 两 种 不 同 的 方法 来 讲解 数据 库 及 数据 表 的 
创建 过 程 。 
1. 在 命令 提示 符 下 创建 数据 库 及 数据 表 
在 数据 库 脚本 文件 导出 后 ， 就 可 以 根据 该 脚本 文件 在 MySQL 中 创建 数据 库 及 数据 表 了 。 具 体 步 
又 如 下 : 
(1) 选择 “开始 ”/“ 所 有 程序 ”/Appserv/MySQL Command Line Client 命令 ， 进 入 到 MySQL 的 
客户 端 命令 行 窗口 ， 输 入 密码 并 按 Enter 键 后 ， 即 可 使 用 MySQL Client 连接 MySQL 数据 库 。 
(2) 在 mysql> 提 示 符 后 面 输 入 以 下 命令 创建 数据 库 db_shop。 
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create database db_shop:; 


(3) 调用 数据 库 db_shop。 
use db_shop: 
(4) 在 数据 库 db_shop 中 创建 一 个 简单 的 数据 表 tb_admin， 包 括 id、name、pwd 等 字段 ， 代 码 
如 下 了 
mysql> create table tb_admin( 
—>idint AUTO INCREMENT PRIMARY KEY, 
-> name varchar(13) NOT NULL. 


->pwd varchar(50) NOT NULL. 
Query OK, 0 rows affected (0.03 sec) 


总 技巧 : 在 输入 SQL 语句 时 ,可 以 一 行 全 部 输出 ,也 可 以 每 个 字段 都 换行 输出 ， 这 里 建议 换行 输出 ， 
这 样 看 上 去 美观 ， 且 吻 懂 ， 在 语句 出 现 错误 时 更 容易 查找 。 
2. 在 phpMyAdmin 图 形 化 工具 下 创建 数据 库 和 数据 表 
在 phpMyAdmin 的 主 界面 中 有 两 个 文本 框 和 一 个 “创建 ”按钮 ， 首 先 在 文本 框 中 输入 数据 库 的 名 
称 ， 然 后 选择 编码 ， 最 后 单 击 “ 创 建 ” 按 钮 ， 这 样 新 的 数据 库 就 可 以 被 创建 成 。 例 如 ， 创 建 一 个 名 称 
为 db_shop 的 数据 库 ， 首 先 在 文本 框 中 输入 db_shop， 之 后 在 下 拉 列 表 框 中 找到 要 使 用 的 编码 ， 在 
Windows 下 一 般 选 择 gb2312_chinese_ci， 如 图 4.16 所 示 。 


] localhost 


加 服务 器 版 本 : 5.0.24a-community-nt-log 
pap ， Protocol version: 10 


回 服 务 器 : localhost via TCP/IP 
_ 加 回国 回回 》 用户 : root@localhost 


数据 库 国 MysQL 字符 集 : UTF.8 Unicode (utfe) 
0 - 贺 WysQL 连接 校对 : [967312_chinese_c 司 @ 


| 输入 雪 据 库 名 | eu sp (Gmesee)” 习 网 时 | 


国 显 示 MySQL 的 系统 变量 四 | 选择 编码 格式 
图 4.16 phpMyAdmin 管理 界面 
最 后 单 击 “ 创 建 ” 按 扭 ， 这 样 名 为 db_shop 的 数据 库 就 被 创建 成 功 。 执 行 结果 如 图 4.17 所 示 。 
先 在 数据 库 db_shop 中 创建 一 个 新 表 


Numberoffields{(3 € ) 
输入 新 表 字 段 个 数 


图 4.17 数据 库 的 建立 


在 右 侧 界面 还 可 以 执行 创建 数据 表 的 操作 ， 首 先 在 表单 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 
“执行 ”按钮 ， 进 入 各 个 字段 的 详细 信息 录入 表单 ， 包 括 字 段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 
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额外 和 索引 的 类 型 等 。 在 这 里 就 完成 了 对 表 结 构 的 详细 设置 ， 如 图 4.18 所 示 。 
图 服务 器 : localhost ， 电 数据 库 : db_shop * 国 表 :tb_admin 


字 良 类 型 加 长 度 导 1 整理 an 本 外 加 图 
mr leachinese ct  H) [nom] lao nement] c 
[name JwRcmR 可 Js [gv2312_chinese_c 司 non | Fes|s 
wd WARcHAR 本 [50 [gb2312_cninese-cr ” 司 [nanw 司 「 Tei 

保存 | 或 应 加 [字段 二 生 


图 4.18 创建 表 信息 


技巧 : 创建 MySQL 数据 库 的 方法 主要 有 两 种 : 一 种 方法 是 在 命令 提示 符 下 通过 输入 SQL 语句 创 
建 ， 但 利用 这 种 方法 效率 相对 较 低 ， 对 SQL 语句 不 是 特别 熟悉 的 初学 者 来 说 也 是 非常 困难 
的 。 另 一 种 是 应 用 数据 库 图 形 化 管理 工具 phpMyAdmin 创建 ， 后面 这 种 方法 操作 上 更 直观 、 
更 便捷 ， 利 用 它 可 以 高 效 地 完成 数据 库 的 设计 。 
结合 该 数据 表 的 创建 方法 ， 读 者 可 以 自行 创建 以 下 数据 表 。 数 据 表 的 设计 结构 如 图 4.19 一 图 4.24 
所 示 。 
(1) tb_shangpin〈 商 品 信息 表 ) 
商品 信息 表 主要 用 于 存储 商品 的 基础 信息 。 该 数据 表 结 构 如 图 4.19 所 示 。 


加 服务 器 : localhost 加 数据 库 : db_shop ， 国 表 :tb_shangpin 
本 外 


字段 类 型 整理 Null 说 明 
i int(4) 否 auto_increment 自动 编号 
mingcheng = varchar(25) 。 gb2312_chinese_cl 是 NULL 商品 名 称 
jianjie mediumtext -gb2312_chinese_cl 是 NULL 商品 价格 
addtime varchar(25) gb2312_chinese_cl 是 WULL 入 市 时 间 
dengji varchar(5) 。 gb2312_chinese_cl 是 NULL 商品 等 级 
xinghao varchar(25) ”gb2312_chinese_ci 是 WULL 商品 型 号 
tupian varchar(200) gb2312_chinese_cl 是 WULL 图 片 路 径 
shuliang int(4) 是 WULL 商品 数量 
cishu int(4) 是 NULL 购买 次 数 
tuilian int4) 是 WULL 是 否 推荐 
typeid int(4) 是 NULL 类 lid 
huiyuanjia 。 varchar(25) ”gb2312_chinese_ci 是 WULL 会 员 价 
shichangjia varchar(25) gb2312_chinese_ci 是 WULL 市 场 价 
pinpai varchar(25) 。 gb2312_chinese_cl 是 NULL 商品 曲折 


图 4.19 商品 信息 表 结构 


(2) tb_dingdan (用户 订 单 表 ) 

用 户 订单 表 主 要 用 于 存储 用 户 的 订单 信息 。 该 数据 表 结 构 如 图 4.20 所 示 。 
(3) tb_admin (管理 员 表 ) 

管理 员 表 主要 用 于 存储 管理 员 的 信息 。 该 数据 表 结 构 如 图 4.21 所 示 。 
(4) tb_user 《用户 信息 表 ) 

用 户 信息 表 主 要 用 于 存储 用 户 的 基础 信息 。 该 数据 表 结 构 如 图 4.22 所 示 。 


迄 1 音 
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胃 服务 器 : localhost 》 避 数据 库 : db_shop 》 国 表 :tb_dingdan 


字 及 类 型 Null 轩 认 要 外 说 明 
el int(4) 否 auto_increment ”自动 编号 这 
dingdanhao ” varchart125) gb2312_chinese_ci 是 NULL 订单 号 
Spc varchar(125) gb2312_chinese ci 是 NULL 商品 审 
sic varchar(125) gb2312_chinese_ci 是 WULL 数量 审 
Shouhuoren varchar(25) ”gb2312_chinese_ci 是 NULL 收 货 人 姓名 
SeX varchar(2) 。 gb2312_chinese_ci 是 WULL 收 件 人 性 别 
dizhi varchar(125) gb2312_chinese_ci 是 WULL 送 货 地 址 
youbian varchar(10) ”gb2312_chinese_ci 是 NULL 邮编 
tel varchar(25) ”gb2312_chinese_ci 是 NULL 联系 电话 
email varchar(25) ”gb2312_chinese_ci 是 WULL El 
shff varchar(25) 。 gb2312_chinese_ci 是 NULL 收 车 方式 
zm varchar(25) 。 gb2312_chinese_ci 是 WULL 嘉 付 方式 
leaveword 。 mediumtedt gb2312_chinese_ci 是 NULL 用 户 留言 
time varchar(25) ”gb2312_chinese_ci 是 NULL 下 单 时 间 
yxiadanren 。 varchar(25) 。 gb2312_chinese_cl 是 NULL 下 单 人 姓名 
A varchar(50) ”gb2312_chinese_ci 是 NULL 订单 状态 
+total varchar(25) 。 gb2312_chinese_ci 是 NULL 价格 总 计 


图 4.20 用 户 订单 表 结构 
困 服务 器 ; localhost 加 数据 库 : db_shop * 国 表 :tb_admin 


字 及 类 型 整理 Nul 默认 额外 说 明 

[a Int(4) 理 auto_increment ”自动 编号 id 

name varchar(13) gb2312_chinese_ci 是 NULL 管理 员 名 

pwd 。 varchar(50) gb2312_chinese_cl 是 NULL 管理 员 密码 

图 4.21 管理 员 表 结构 

胃 服务 器 : localhost ， 数据 库 : db_shop ， 国 表 :tb_user 

字段 整理 Nul 默认 额外 说 明 
a int(4) 理 auto_increment 自动 编号 这 
name varchar(25) ”gb2312_chinese_ci 是 NULL 用 户 名 
pwd varchar(50) ”gb2312_chinese_cl 是 NULL 用 户 窗 码 
dongjie int(4) 是 NULL 标记 用 户 是 否 被 沫 结 
emall varchar(25) 。 gb2312_chinese_cl 是 NULL 用 户 E_mail 地 址 
sfzh varchar(25) gb2312_chinese_cl 是 NULL 用 户 身份 证 号 
tel varchar(25) ”gb2312_chinese_cl 是 WULL 联系 电话 
tishi varchar(50) ”gb2312_chinese_ci 是 NULL 窗 码 找 回 提示 
huida varchar(50) ”gb2312_chinese_ci 是 NULL 密码 找 回答 案 
dizhi varchar(100) gb2312_chinese_cl 是 WULL 用 户 联系 地 址 
youbian varchar(25) ”gb2312_chinese_cl 是 NULL 用 户 邮编 
regtime varchar(25) 。 gb2312_chinese_ci 是 NULL 用 户 注册 时 间 
truename varchar(25) ”gb2312_chinese_ci 是 NULL 用 户 真实 姓名 
pwd1 varchar(50) ”gb2312_chinese_ci 是 NULL 未 加 密 的 用 户 密 码 
qq varchar(25) ”gb2312_chinese_ci 是 NULL 用 户 QQS 玛 


(5) tb leaveword〈 用 户 留 言 表 ) 


图 4.22 ”用户 信 息 表 结构 


用 户 留 言 表 主要 用 于 存储 用 户 留言 的 相关 信息 。 该 数据 表 结 构 如 图 4.23 所 示 。 


(6) tb pingjia〈 商 品评 价 表 ) 


商品 评价 表 主 要 用 于 存储 用 户 对 商品 的 评论 信息 。 该 数据 表 结 构 如 图 4.24 所 示 。 
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胃 服务 器 : localhost ， 电 数据 库 : db_shop 》 国 表 : tb_leaveword 

字段 关 型 整理 [7 杞 外 说 明 
int(4) auto_increment ”自动 篇 号 过 
varchar(100) gb2312_chinese_ci 
text gb2312_chinese_ci 留言 时 间 


varchar(25) 。 gb2312_chinese_ci 是 留言 内 容 


留言 主题 


否 
int(4) 是 NULL 用 户 ia 
是 


图 4.23 用 户 留 言 表 结构 


困 服务 器 : localhost ， 成 数据 库 : db_shop ， 国 表 : tb_pingjia 
字 及 类 型 Null -默认 贪 外 


说 明 
过 int(4) 理 auto_increment 自动 编号 这 
userid int(4) 是 WULL 用 户 这 
spid int(4) 是 NULL 商品 和 
te varchar(100) gb2312_chinese_cl 是 NULL 评价 主题 
content text gb2312_chinese_ci 是 NULL 评价 内 容 
tme varchar(25) gb2312_chinese_ci 是 NULL 评价 时 间 


图 4.24 商品 评价 表 结 构 
[加 说 明 : 限于 篇 幅 ， 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 可 参见 本 书 附带 的 光盘 。 


4.$ 公共 模块 设计 


4.5.1 数据 库 连 接 文件 


这 里 进行 的 第 1 项 内 容 就 是 建立 与 数据 库 的 连接 文件 conn.php。 数 据 库 连 接 文件 在 以 后 的 其 他 动 
态 页 中 均 要 涉及 到 ， 所 以 笔者 把 涉及 到 的 脚本 文件 放 在 了 这 里 进行 重点 介绍 。 以 后 再 涉及 数据 库 连接 
文件 时 就 不 再 袭 述 了 。conn.php 文件 的 代码 如 下 : 


例 程 01 代码 位 置 : 光盘 \TM\04\shop\conn\conn.php 


<?php 

© Slink=mysql_connect('localhost','root','root’): // 连 接 到 MySQL 服 务 器 

@ mysql_select_db('db_shop',$link): // 指 定 连接 到 名 称 为 db_shop 的 数据 库 
?> 

< 代码 贴 填 


@ mysql conect0: 连接 MySQL 服务 器 ， 服 务 器 的 用 户 名 为 root， 密 码 为 Toot。 
@ mysql_ select_ db0: 用 于 连接 指定 的 MySQL 数据 库 。 


如 果 某 个 页 面 中 需要 进行 数据 库 的 操作 ， 在 页 面 的 前 台 直 接 包含 该 文件 即 可 ， 代 码 如 下 : 


<2php 
include ("conn/conn.php"): // 包 含 数据 库 文件 
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4.5.2 CSS 样式 表 文件 


CSS 是 Cascading Style Sheets( 层 又 样式 表单 ) 的 简称 ， 是 一 种 简单 、 灵 活 、 易 学 的 工具 ， 可 使 任 
何 浏览 器 都 听从 指令 ， 知 道 该 如 何 显 示 元 素 及 其 内 容 。 人 掌握 CSS 样式 表 不 仅 能 更 好 、 更 快 地 完成 网 页 
设计 使 页 面具 有 动态 效果 ， 还 有 助 于 统一 网 站 的 整体 风格 。 

在 网 页 中 使 用 CSS 的 方法 如 下 : 

回 ”把 CSS 文档 放 到 <head></head> 标 记 中 。 

<head><style type="text/css"> … </style> </head> 


了 把 CSS 样式 表 写 在 HIML 行内 ， 代 码 如 下 : 
<p style="font-size: 14pt; color: blue”> 蓝 色 14 号 文字 </p> 


这 是 采用 <style=" "> 的 格式 把 样式 写 在 HTML 中 的 任意 行内 ， 这 种 方法 比较 方便 灵活 。 

回 ”把 编辑 好 的 CSS 文档 保存 成 扩展 名 为 “.css” 的 外 部 文件 ， 然 后 在 <head> 标 记 中 调用 该 文件 ， 
调用 方法 的 代码 如 下 : 

<head> <link rel=stylesheet type="text/css" href=".css 文 档 的 相对 路 径 "> … </head> 


这 种 方式 能 使 多 个 文档 同时 使 用 相同 的 样式 ， 从 而 能 够 减少 大 量 的 元 余 代码 。 
电子 商务 平台 采用 <link> 将 扩展 名 为 “.css” 的 外 部 文件 嵌入 到 网 页 中 ， 代 码 如 下 : 


<link href="css/font.css" rel="stylesheet"> 
电子 商务 平台 中 应 用 的 font.css 样式 表 文件 的 代码 如 下 : 


例 程 02 ”代码 位 置 : 光盘 \TMVWO4\ shop\css\font.css 
<!-- 定 义 文字 链接 的 CSS 样 式 --> 
A:link { 

COLOR: #006699: TEXT-DECORATION: none 
» 
Aivisited { 

COLOR: #006699: TEXT-DECORATION: none 
} 
Ai:active{ 

COLOR: #006699:; TEXT-DECORATION: none 
} 
A:hover { 

COLOR: #000000 


} 

<!-- 定 义 网 页 背景 及 文字 颜色 --> 

BODY { 
background-image: url(../images/bg.gif): 
margin-top: Opx: 


b 
<!-- 定 义 表格 文字 的 打下 和 颜色 --> 


TID,TH{ 
FONT-SIZE:12px: COLOR: #006699: 
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» 
<!-- 定 义 按 扭 的 CSS 样 式 --> 


.buttoncss { 


font-family: "Tahoma", "宋体 ": 
font-size: Ppt: color: #003399; 

border: 1px #003399 solid: 
color:006699: 

BORDER-BOTTOM: #93bee2 1px solid; 
BORDER-LEFT: #93bee2 1px solid: 
BORDER-RIGHT: #93bee2 1px solid; 
BORDER-TOP: #93bee2 1px solid; 
background-color: #e8f4f: 

CURSOR: hand: 

font-style: normal ; 


} 
<!-- 定 义 输入 框 的 CSS 样 式 --> 
.inputcss { 


font-size: pt; 

color: #003399; 

font-family: "宋体 "; 

font-style: normal: 

border-color: #93BEE?2 #93BEE?2 #93BEE?2 #93BEE?2 : 
border: 1px 加 3BEE2 solid: 


} 
<!-- 定 义 无 边 框 的 输入 框 的 CSS 样 式 --> 


.inputcssnull { 


font-size: 9pt; 

color: #003399; 
font-family: "宋体 ": 
font-style: normal; 

border: Opx #93BEE?2 solid: 


} 
<!-- 定 义 滚动 条 的 CSS 样 式 --> 
.scrollbar{ 


SCROLLBAR-FACE-COLOR: #FFDD22; 
FONT-SIZE: pt: 
SCROLLBAR-HIGHLIGHT-COLOR: #69BC2C: 
SCROLLBAR-SHADOW-COLOR: #69BC2C: 
SCROLLBAR-3DLIGHT-COLOR: #69BC2C: 
SCROLLBAR-ARROW-COLOR: #FFFFE: 
SCROLLBAR-TRACK-COLOR: #69BC2C: 
SCROLLBAR-DARKSHADOW-COLOR: #69BC2C 


4.6 前 侣 首页 设计 


在 无 数 个 相互 竞争 的 网 页 中 ， 特 别 是 对 电子 商务 网 站 来 说 ， 首 页 极为 


要 ， 它 必须 展现 网 站 的 特 
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性 ， 并 积极 地 加 以 表现 ， 首 页 设计 的 好 坏 将 直接 影响 顾客 的 购买 欲望 和 情绪 。 在 电子 商城 的 首页 设计 
中 ， 首 先 必 须 把 商城 推出 的 最 新 商品 、 推 荐 商品 、 热 门 商品 、 最 新 公告 等 商城 的 特色 和 动态 信息 展现 
给 顾客 ， 然 后 再 提供 查看 订单 、 购 物 车 、 商 品 分 类 查询 等 业务 。 


4.6.1 前 台 首 页 概述 


网 站 首页 是 关于 网 站 的 建设 及 形象 宣传 ， 它 对 网 站 生存 和 发 展 起 着 非常 重要 的 作用 。 首 页 设计 的 
好 坏 将 直接 影响 到 顾客 的 购买 欲望 ， 同 时 ， 首 页 也 是 一 个 信息 含量 较 高 ， 内 容 丰 富 的 宣传 平台 ， 用 户 
不 但 可 以 在 第 一 时 间 内 掌握 商城 最 新 商品 、 热 门 商品 、 推 荐 商品 ， 还 可 以 实现 商品 搜索 、 订 单 查询 等 
功能 。 电 子 商务 平台 前 台 首 页 主要 包含 以 下 内 容 : 
网 站 菜单 导航 : 主要 包括 最 新 商品 、 推 荐 商品 、 热 门 商 品 、 商 品 分 类 、 用 户 中心 、 订 单 查 询 、 
我 的 购物 车 。 
商品 信息 的 快速 搜索 及 高 级 搜索 引擎 。 
最 新 商品 模块 : 主要 按时 间 先 后 顺序 展示 最 新 商品 及 详细 信息 查看 。 
推荐 商品 模块 : 主要 展示 商城 重点 推荐 的 商品 及 详细 信息 查看 。 
热门 商品 模块 : 主要 用 于 展示 销量 最 高 的 商品 及 详细 信息 查看 。 
会 员 登录 模块 : 主要 用 于 会 员 登录 ， 登 录 后 可 以 购买 商品 、 查 询 提 交 的 订单 、 查 看 个 人 消费 
情况 。 
用 户 中 心 模块 : 主要 用 于 在 用 户 登 录 后 ， 对 个 人 资料 的 修改 、 对 登录 密码 的 修改 、 用 户 留言 
和 评论 信息 。 

回 ”站 内 公告 模块 : 主要 用 于 发 布 网 站 提供 部 分 商品 信息 以 及 购买 商品 的 优惠 制度 等 信息 。 

回 ”友情 链接 模块 : 主要 用 于 企业 链接 的 广告 信息 。 

电子 商务 平台 前 台 首页 分 为 网 站 导航 、 版 权 信息 、 左 分 栏 、 内 容 分 栏 4 个 部 分 。 下 面 看 一 下 本 案 
例 中 提供 的 商城 前 合 首页， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\04\shop\index.php， 如 图 4.25 所 示 。 

电子 商务 网 站 的 设计 是 展现 企业 形象 、 介 绍 商 品 和 服务 、 体 现 企 业 发 展 战略 的 重要 途径 ， 这 也 是 
开发 网 站 的 根本 目的 ， 因 此 必须 首先 明确 用 户 的 需求 ， 从 而 做 出 切实 可 行 的 设计 计划 。 要 根据 消费 者 
的 需求 、 市 场 的 状况 、 企 业 自身 的 情况 等 进行 综合 分 析 ， 牢 记 以 “消费 者 (customer) ”为 中 心 ， 而 不 
是 以 “美术 ”为 中 心 进行 设计 规划 。 不 同 的 产品 网 站 所 要 表达 的 信息 类 型 也 不 一 样 ， 要 用 适合 表达 其 
产品 特点 的 风格 和 色彩 的 设计 。 

电子 商务 平台 采用 二 分 栏 结构 布局 ， 具 有 页 面 简 练 、 大 气 、 个 性 鲜明 等 特点 ， 从 而 体现 电子 商务 
网 站 的 特色 和 个 性 化 ， 网 站 示意 图 如 图 4.26 所 示 。 


4.6.2 ”前 台 首页 技术 分 析 


同 罗 网 网 加 加 


加 


为 了 保证 页 面 的 整洁 和 增强 页 面 的 可 维护 性 ， 在 前 台 首 页 面 中 使 用 引用 语句 来 包含 主要 的 功能 页 
面 ， 各 个 功能 模块 分 别 保存 在 单独 的 文件 中 。 这 样 做 的 目的 是 使 系统 具有 统一 的 风格 ， 并 且 如 果 对 某 
项 功能 进行 维护 ， 只 需要 修改 ttpphp、leftphp、index.php 或 bottom.php 页 即 可 ， 不 需要 每 页 都 进行 改 
动 ， 这 样 可 以 很 大 程度 地 提高 网 站 开发 效率 ， 并 且 维护 起 来 也 是 非常 方便 的 。 
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EE so 


a 网 站 导航 、LOGO、 搜 索引 擎 


用 尸 : mr 

害 码 : 0000 

验证 : R2058 2058 
提交 | 注册 找 回 密码 


* 程序 正在 测试 中 
rE] 
* 本 商城 从 即日 起 推出 忧 囊 活 

* 所 有 会 员 丙 品 一 律 15 折 忧 圳 

* 噶 子 商务 系统 全 面 推出 着 澳 


如 1 CopyRishts reserved 2007 吉林 省 ra 科技 有 限 公司 
堵 服 热线 : 0431-B19Tesys 了 -mail : rongzra8sina con 


图 4.25 电子 商务 平台 前 台 首 页 


bottom php 


图 4.26 电子 商务 网 站 示意 图 
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在 PHP 页 面 中 包含 文件 有 4 种 方法 : 

应 用 include 语句 实现 。 

使 用 include0 语 句 包含 外 部 文件 时 ， 只 有 代码 执行 到 include 语句 时 才 将 外 部 文件 包含 进来 ， 当 所 
包含 的 外 部 文件 发 生 错误 时 ， 系 统 只 给 出 一 个 警告 ， 而 整个 php 文件 则 继续 向 下 执行 。include 语句 的 
语法 如 下 : 

void include(string filename): 

参数 flename 是 指定 的 完整 路 径 的 文件 名 。 


全 注意 : 从 PHP 5.2 开始 ，include 语句 将 会 比 include once 语句 执行 速度 快 ， 因 为 include 语句 可 以 
优化 脚本 ， 不 会 检查 包括 的 文件 或 函数 是 否 已 经 存在 。 

应 用 require 语句 实现 。 

require 语句 与 include 语句 类 似 ， 都 是 实现 对 外 部 文件 的 调用 。 当 使 用 require 语句 载 入 文件 时 ， 
它 会 作为 PHP 文件 的 一 部 分 被 执行 ， 语 法 如 下 : 

void require(string filename): 

参数 flename 是 指定 的 完整 路 径 文件 名 。 

下 面 应 用 require 语句 包含 并 运行 指定 的 外 部 文件 top.php， 代 码 如 下 : 

<?php require("top.php"): /嵌入 外 部 文件 top.php 页 ?> 

回 ”应 用 include_once 语句 实现 。 

应 用 include_once 语句 多 次 调用 相同 的 文件 时 ， 程 序 只 会 调用 一 次 。 例 如 ， 要 导入 的 文件 中 存在 
一 些 自 定 义 函 数 ， 那 么 如 果 在 同一 个 程序 中 重复 导入 这 个 文件 ， 在 第 2 次 导入 时 便 会 发 生 错误 ， 因 为 
PHP 不 允许 相同 名 称 的 函数 被 重复 声明 两 次 ， 语 法 如 下 : 

Void include_once (string filename): 

filename 参数 是 指定 的 完整 路 径 的 文件 名 。 

下 面 应 用 include_once 语句 包含 并 运行 指定 的 外 部 文件 top.php， 代 码 如 下 : 

<?php include_once("top .php"): 1/ 嵌入 外 部 文件 top.php 页 ?> 

应 用 require_once 语句 实现 。 

require_once 语句 是 require 语句 的 延伸 , 它 的 功能 与 require 语句 基本 类 似 , 不 同 的 是 , require_once 
语句 会 先 检查 要 导入 的 文件 是 不 是 已 经 在 该 程序 中 的 其 他 地 方 被 调用 过 ， 如 果 是 的 话 就 不 会 再 次 重复 
调用 该 文件 ， 语 法 如 下 : 

Void require_once (string filename): 

filename 参数 是 指定 的 完整 路 径 的 文件 名 。 

下 面 应 用 require_once 语句 包含 并 运行 指定 的 外 部 文件 top.php， 代 码 如 下 : 

<?php require_once("top.php"): /| 嵌入 外 部 文件 top.php 页 ?> 
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4.6.3 ”前台 首页 的 实现 过 程 


国 前 台 首 页 使 用 的 数据 表 : tb_shangpin、tb_user、tb_gonggao、tb links 
在 一 个 网 站 中 ， 前 台 首页 被 访问 的 次 数 是 比较 多 的 。 为 了 加 快 页 面 的 运行 速度 、 提 高 访问 量 ， 本 
实例 前 台 首 页 使 用 include 语句 包含 主要 功能 模块 ， 代 码 如 下 : 


例 程 03 ”代码 位 置 : 光盘 \TM\04\ shop\index.php 
<table width="766" border="0" cellspacing="0" cellpadding="0"> 
<t> 
<td colspan="2"> 
oo <?php include("top.php"):?> 
</td> 
</tr> 
<t> 
<td width="209"> 
© <?php include("left.php");?> 
<td> 
3 <! 一 商品 展示 模块 的 代码 部 分 --> 
<td> 
和 /商品 展示 模块 代码 部 分 略 


和 > 
<td colspan="2"> 
9 <?php include("bottom.php"); ?> 
</td> 
</t> 
</table> 


< 人 代码 贴 十 
@ 应 用 include 语句 包含 top.php 文件 ， 该 文件 用 于 显示 网 站 导航 、 企 业 LOGO、 搜 索引 擎 及 当前 登录 的 会 员 名 称 。 
四 应 用 include 语句 包含 leftphp 文件 ， 该 文件 用 于 显示 用 户 登 录 、 商 城 公告 及 友情 链接 信息 。 
@ 在 前 台 首 页 (index php ) 中 ， 应 用 表格 布局 的 方式 展示 商品 信息 。 
@ 应 用 include 语句 包含 bottom php 文件 ， 该 文件 用 于 显示 版 权 信息 。 


4.7 商品 展示 模块 设计 


本 系统 在 前 台 为 用 户 提供 了 不 同 的 商品 展示 方式 ， 从 而 便于 消费 者 了 解 市 场 行情 ， 能 够 使 消费 者 
有 目的 地 选 购 一 些 商品 。 
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4.7.1 商品 展示 模块 概述 


商品 展示 模块 属于 电子 商务 平台 的 子 页 ， 主 要 显示 电子 商务 平台 的 商品 信息 。 商 品 信息 主要 为 用 
户 提供 信息 资源 ， 是 决定 用 户 是 否 购买 商品 的 主要 因素 ， 因 此 商品 展示 模块 对 电子 商务 平台 来 说 尤为 
重要 。 商 品 展示 模块 主要 实现 以 下 功能 : 
商品 的 分 类 信息 展示 。 
分 页 显示 相应 类 别 的 商品 信息 。 
用 户 可 直接 进行 商品 的 购买 。 
显示 重点 推荐 商品 、 最 新 商品 及 热门 商品 信息 。 


网 罗网 加 


4.7.2 商品 展示 模块 技术 分 析 


在 电子 商务 平台 中 ， 考 虑 到 商品 的 种 类 会 非常 多 ， 因 此 采取 分 页 的 方式 进行 显示 。 本 购物 系统 中 
很 多 模块 多 采用 了 这 种 技术 。 读者 只 要 熟悉 商品 分 类 模块 的 分 页 实现 方法 就 可 以 将 该 方法 应 用 到 它 处 ， 
触 类 旁 通 。 

单 击 商品 分 类 超 链接 ， 即 可 以 分 页 的 形式 查看 该 类 别 下 所 有 商品 的 详细 信息 ， 代 码 如 下 : 

例 程 04 ”代码 位 置 : 光盘 \TM\04\shop\showfenlei.php 


<?php 
$sql=mysql_ query("select count(*) as total from tb_shangpin where typeid="".Sid."" order by addtime desc ",$conn); 
Sinfo=mysql_ fetch_array($sql): // 检 索 指 定 商品 类 别 下 的 所 有 商品 信息 
S$total=$info[total]: // 计 算 该 类 别 下 所 有 商品 的 总 和 
这 $total 一 0) { /如果 商 品 总 和 为 0， 则 说 明 该 类 暂 无 商品 ， 给 出 相关 提示 
echo "<div align='center> 本 站 暂 无 该 类 产品 !</div>": 
} 
else { // 否 则 ， 以 分 页 形式 输出 商品 信息 
Spagesize=3; // 每 页 显示 3 条 商品 信息 
if ($total<=$pagesize){ /如 果 商 品 总 数 小 于 每 页 最 多 显示 的 页 数 则 总 页 数 应 为 1 
S$pagecount=1; 


} 
// 如 果 总 商品 数 不 能 整除 每 页 最 多 显示 的 商品 数 ， 则 总 页 数 应 该 比 两 者 整除 之 商 多 1 


这 (Stotal%Spagesize)!=0){ 
S$pagecount=intval($total/$pagesize)+1: 

}else{ // 如 果 总 商品 数 能 整除 每 页 最 多 显示 的 商品 数 ， 则 总 页 数 应 为 两 者 之 商 
Spagecount=$total/$pagesize; 

由 

if(($_GET[page])—""){ // 如 果 $_GET[page] 的 值 为 空 ， 则 使 默认 显示 的 页 为 第 1 页 
Spage=1: 

}else{ // 否 则 ， 使 当前 显示 的 页 码 为 获取 的 Spage 的 值 


Spage=intval($_ GET[page]): 
B 
$sql1=mysql query("select * 他 om tb_shangpin where typeid=".S$id." order by addtime desc limit 
".(Spage-1)*Spagesize.",Spagesize ".$conn): // 实 现 分 页 显示 
while($infol=mysql fetch array($sqll)) { // 应 用 while 循 环 语句 输出 商品 的 详细 信息 
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?> 
/商品 详细 信息 展示 的 代码 略 
<table width="550" height="25" border="0" align="center" cellpaddine="0" cellspacine="0"> 
<tr> 
<td><div align="right"> &nbsp: 本 站 共有 该 类 商品 &nbsp: 
<?php 
echo Stotal: // 输 出 该 类 别 下 的 商品 总 数量 
?> 


&nbsp; 件 &nbsp: 每 页 显示 &nbsp:<?php echo $pagesize:?>&nbsp: 件 &nbsp: 第 &nbsp:<?php echo $Spage;?>&nbsp; 页 / 共 
&nbsp:<?php echo $pagecount: ?>&nbsp: 页 


<?php 
if($page>=2){ // 如 果 当 前 页 码 大 于 等 于 2， 则 显示 首页 及 前 一 页 链接 
到 
by <a href="showfenlei.php?id=<?php echo $id:?>&page=1" title=" 首 页 "><font face="webdings"> 9 </font></a> 
@ <a href="showfenlei.php?id=<?php echo $id:?>&page=<?php echo $Spage-1:?>" title=" 前 一 页 "><font 
face="webdings"> 7 </font></a> 
<?php 
} | 
if($pagecount<=4){ /如 果 总 页 数 小 于 或 等 于 4， 则 显示 所 有 页 的 链接 
for($i=1;$i<=$pagecount:$i++){ 
?> 
<a href="showfenlei.php?id=<?php echo $id:?>&page=<?php echo $i:?>"><?php echo $i:?></a> 
<?php 
} = Ss 
}else{ /如 果 总 页 数 大 于 4， 则 只 显示 前 4 页 链接 ， 并 显示 尾 页 和 后 一 页 链接 
for($i=1:$i<=4:$i+H){ 
?> 
<a href="showfenlei.php?id=<?php echo $id;?>&page=<?php echo $i:?>"><?php echo $i:7?></a> 
<?php }?> 
© <a href="showfenlei.php?id=<?php echo $id;:?>&page=<?php echo Spage-1:2>" title=" 后 一 页 "><font 
face="webdings"> 8 </font></a> 
9 <a href="showfenlei.php?id=<?php echo $id:?>&page=<?php echo Spagecount:?>" title=" 尾 页 "><font 
face="webdings"> : </font></a> 
<?php 
} 
We 
人 代码 贴 十 


@ <font face="webdings"> 9 </font>: 输出 “首页 ”文字 标识 4 。 
@ <font face="webdings"> 7 </font>: 输出 “前 一 页 ”文字 标识 4。 
@ <font face="webdings"> 8 </font>: 输出 “下 一 页 ”文字 标识 惫 。 
@ <font face="webdings"> :</font>: 输出 “ 尾 页 ”文字 标识 出 。 


总 技巧， 为 了 加 强 网 页 的 可 操作 性 ， 商 品 展示 页 面 实现 了 分 页 功能 ， 并 通过 设计 文字 的 特殊 字体 
webdings 输出 首页 、 前 一 页 、 下 一 页 、 尾 页 的 图 标 (在 指定 的 标准 页 范围 显示 ， 详 见 代码 
注释 ) 。 
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4.7.3 商品 分 类 展示 的 实现 过 程 


国 。 商品 分 类 模块 使 用 的 数据 表 : tb_shangpin、tb_type 

在 电子 商务 平台 首页 中 ， 设 置 商品 分 类 展示 不 仅 可 使 电子 商务 平台 的 所 有 商品 分 门 别 类 地 显示 出 
来 ， 而 且 为 用 户 选择 商品 提供 了 很 大 的 方便 。 

在 网 站 功能 导航 栏 中 单 击 “ 商 品 分 类 ” 超 链 接 ， 进 入 到 商品 分 类 展示 页 面 。 在 该 页 面 中 系统 自动 
检索 出 所 有 的 商品 分 类 超 链接 ， 单 击 “ 家 居 日 用 ” 超 链接 ， 将 输出 该 类 别 下 的 所 有 商品 信息 ， 运 行 结 
果 如 图 4.27 所 示 。 


计算 机 图 书 玩具 类 网 络 莫 戏 服饰 家 居 日 用 电影 音乐 数码 产品 通讯 设备 电脑 软件 
ee 
di 一 商品 :44 商品 型 号 : 301X 
证 : 了 8494 人 无 
| | 提 变 | 注册 找 回 码 上 市 日 央 : 。 2005-7-8 制作 数量 : 8000 商品 等 级 : 一 般 
商场 价 : 。 1600 元 会 员 价 : 1699 元 折扣 ;95% 
商品 名 称 : 。 水 杯 
ad Na: 44 商品 型 号 : 4 
* 再 天 的 价格 已 经 打 啊 ) 。 商品 简介 : 无 
。 本 两 城 人 即日 起 推出 优惠 活 上 市 日 央 : 199s-1-1 制作 数量 - 16o0 商品 等 级: 精品 
所 有 会 员 商 昌 一 香 15 折 习 吉 bi 人 元; ee 
， 电子 商务 系统 全 面 推出 辐 内 | RY Ws 
rm 本 站 共有 该 闫 商 品 2 件 等 页 显示 3 件 第 1 页 / 共 ! 页 1 


图 4.27 商品 分 类 展示 页 面 的 运行 结果 


首先 建立 一 个 单独 的 tb_type 表 用 来 存储 商品 大 类 ， 然 后 通过 do…while 循环 语句 把 这 些 记 录 的 
typename 字段 〈 商 品类 别名 称 ) 都 显示 出 来 ， 并 且 每 个 商品 类 别名 称 都 设 有 超 链接 ， 单 击 该 超 链接 用 
户 可 以 查看 该 类 别 下 所 有 商品 的 详细 信息 ， 代 码 如 下 : 


例 程 05 ”代码 位 置 : 光盘 \TM\04\shop\showfenlei.php 


<?php 
$sql=mysql query("select * from tb_type order by id desc".Sconn): 
@ S$info=mysql_fetch_object(SsqD: /查询 商品 类 别 信息 表 中 的 信息 
这 $info 一 false){ // 如 果 查 询 结果 为 假 ， 则 弹出 相关 的 提示 信息 
echo "本 站 暂 无 商品 !": 
y 
else { // 否 则 ， 输 出 商品 类 别 信息 
dof // 应 用 循环 语句 为 商品 类 别名 称 添加 超 链接 
@ echo "<a href='showfenlei.php?id=".$info->id.">".S$info->typename."&nbsp:</a>": 
}while(Sinfo=mysql fetch_object($sqD)); 
?> 


q 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


志和 代码 贴 十 
@ mysql fetch_object: 从 结果 集中 获取 一 行 作为 对 象 ， 如 果 没有 更 多 行 ， 则 返回 false。 
@ $info->typename: 从 结果 集中 获取 商品 类 型 名 称 。 


4.7.4 最 新 商品 展示 的 实现 过 程 


国 ”最 新 商品 模块 使 用 的 数据 表 : tb_shangpin 
在 网 站 功能 导航 栏 中 单 击 “最 新 商品 ” 超 链接 ， 进 入 到 最 新 商品 展示 页 面 。 在 该 页 面 中 系统 显示 
按 管理 员 发 布 商品 的 时 间 降 序 排列 的 前 4 件 商品 ， 运 行 结果 如 图 4.28 所 示 。 


i aas eR 
可 | ee 
> 2 商品 BR; 时 沿 商品 型 号 ; mm-so0 
ME! ] 商品 简介 ; 。 引 e 时 尚 的 数码 科技 
| 提交 | 注册 找 回 密 码 上 市 日期 2006-55 币 人 数量 : 20000 。。 商品 等 级 ; 精品 
一 商场 价 : 。 2699 元 会 员 价 : 2599 元 折扣 ;97% 
po 0 i 
。 程 序 正在 全 坛 中 商品 名称: 全 目 动 夺 衣 机 
商品 品牌 ;144 商品 型 号 : 301X 
* 商城 的 价格 已经 亲 啊 一 rp 
* 本 商 枯 从 芭 日 起 失 出 优 训 括 上 币 晶 央 ;。 2005-7-8 制作 数量 : 5000 商品 等 级; 一 般 
。 所 有 会 肌 商 品 一 律 75 折 居 吉 商场 价 : 。 1800 元 会 员 价 : 1899 元 折扣 :5% 
* 电子 商务 系统 全 面 推出 遇 


4.28 最 新 商品 展示 页 面 的 运行 结果 


在 商品 信息 表 tb_shangpin 中 开辟 一 个 addtime 字段 ， 应 用 该 字段 记录 商品 的 添加 时 间 ， 在 前 台 显 
示 商 品 时 只 需 应 用 这 个 字段 将 所 有 商品 降序 排列 , 然后 应 用 do…while 循环 语句 将 排 好 序 的 记录 中 的 前 
4 条 记录 输出 到 浏览 器 即 可 ， 代 码 如 下 : 

例 程 06 ”代码 位 置 : 光盘 VTMVWO4\ shop\ shownewphp 


<?php 
$sql=mysql_query("select * from tb_shangpin order by addtime desc limit 0,4".Sconn): /查询 最 新 的 4 件 商品 
Sinfo=mysql fetch_array($sq)): 


这 S$info 一 false){ /如果 $info 的 值 为 室 ， 则 说 明 商 品 表 中 无 商品 
echo "本 站 暂 无 最 新 产品 !"; 

BE 

elsef /如 果 $info 的 值 不 为 空 ， 显 示 所 有 商品 信息 
dof /应 用 do…while 循 环 语句 显示 所 有 的 商品 信息 

es 

<t> 


<td width="89" rowspan="6"><div align="center"> 


<?php 
if(Sinfo[tupian]—"){ // 如 果 图 片 为 空 ， 则 给 出 相应 的 提示 信息 


> 
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echo " 暂 无 图 片 ": 
} 
else{ // 否 则 ， 输 出 图 片 信息 
?> 


<a href="lookinfo.php?id=<?php echo Sinfo[id];?>"> <img border="0" src="<?php echo Sinfo[tupian];?>" 
width="80" height="80"></a> 
<?php 
} 
?> 
</div></td> 
<td width="93" height="20"><div align="center" style="color: #000000"> 商 品名 称 : </div></td> 
<td colspan="$5"><div align="left"><a href="lookinfo.php?id=<?php echo S$info[id]:?>"><?php echo 
Sinfo[mingcheng]:?></a></div></td> 
</t> 
/商品 详细 信息 部 分 代码 略 
<?php 
jwhile(Sinfo-mysql_fetch_array(SsqD); 
站 


?> 


名 注意 : 在 do…while 循环 语句 中 ，while 后 的 分 号 不 能 省 略 。 
4.7.5 查看 商品 详细 信息 的 实现 过 程 


国 ”查看 商品 详细 信息 模块 使 用 的 数据 表 : tb_shangpin 

为 了 能 让 客户 全 面 了 解 某 件 商品 ， 本 购物 系统 设置 了 查看 商品 详细 信息 模块 。 在 网 站 首页 商品 展 
示 区 ， 单 击 相应 商品 中 的 “查看 详情 ”按钮 〈 或 在 商品 信息 展示 页 面 ， 单 击 相应 的 商品 名 称 超 链接 ) ， 
即 可 进入 到 “查看 商品 详细 信息 ”页 面 。 商 品 详细 信息 展示 页 面 的 运行 结果 如 图 4.29 所 示 。 


图 4.29 商品 详细 信息 展示 页 面 的 运行 结果 


[0 说 明 : 在 该 模块 中 用 户 不 仅 可 以 全 面 了 解 商 品 信息 ， 而 且 网 站 会 员 还 可 以 发 表 用 户 个 人 评论 ， 根 
据 用 户 的 评论 ， 管 理 人 员 可 以 对 商品 做 一 些 调整 ， 例 如 增加 一 些 口 碑 较 好 的 商品 ， 删 除 一 
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些 评价 相对 较 差 的 商品 。 限 于 篇 幅 ， 用 户 评论 的 功能 本 章 不 做 重点 讲解 ， 详 细 代码 请 读者 
参见 本 书 附 赠 光 盘 。 
在 网 站 首页 商品 展示 区 ， 添 加 “查看 详情 ”按钮 的 代码 如 下 : 
例 程 07 ”代码 位 置 ， 光盘 \TM\04\shop\index php 
<a href="lookinfo.php?id=<?php echo $info[id]:?>"><img src="images/xiangxi btm.gif' width="60" height="18" 
border="0"></a> 
单 击 相 应 商品 中 的 “查看 详情 ”按钮 ， 提 交 商 品 id 到 数据 处 理 页 bookinfo.php 页 ， 通 过 GET 方法 
接收 提交 的 商品 的 id 值 , 然后 应 用 mysql_query0 函 数 检索 该 商品 id 所 对 应 的 商品 信息 ,最 后 在 设计 好 
的 表格 中 通过 echo0 语 句 输出 该 商品 的 详细 信息 ， 代 码 如 下 : 
例 程 08 ”代码 位 置 : 光盘 \TM\04\ shop\ lookinfo.php 


<?php 
include("conn/conn.php"): // 连 接 数 据 库 文件 
S$sql=mysql query("select * from tb shangpin where id=".$ GETI[id]."",Sconn); 
Sinfo=mysql_fetch_object($sqD); /从 商品 表 中 获取 商品 信息 
?> 
<t> 
<td width="89" height="80" rowspan="4" align="center" valign="middle" bgcolor="#FFFFFF"><div align="center"> 
<?php 
if($info->tupian=—=""){ // 如 果 该 商品 没有 图 片 ， 则 弹出 “ 暂 无 图 片 ” 
echo " 暂 无 图 片 "; 
) 
else{ // 如 果 该 商品 有 图 片 ， 则 以 固定 尺寸 显示 
~ 


<a href="<?php echo Sinfo->tupian;?>" target="_blank"><img src="<?php echo Sinfo->tupian:?>" alt=" 查 看 大 图 " 
width="80" height="80" border="0"></a> 
<2php 
bp 
> 
</div></td> 
<td width="92" height="20" align="left" bgcolor="#FFFFFF"> 商 品名 称 : </td> 
<td width="134" bgcolor="#FFFFFF"> &nbsp:<?php echo $info->mingcheng:?> </td> 
<td width="100" bgcolor="#FFFFFF"> 入 市 时 间 : ”</td> 
<td width="129" bgcolor="#FFFFFF">&nbsp:<?php echo $info->addtime:?> </td> 


</t> 
要 /部 分 商品 信息 显示 代码 略 
</table> 


[加 说明: 单 击 图 片 超 链接 ， 将 以 原始 尺寸 显示 图 片 资源 。 
4.7.6 单元 测试 


本 模块 主要 通过 商品 分 类 实现 商品 的 展示 ， 并 对 分 类 的 商品 进行 分 页 处 理 ， 以 此 达到 直观 显示 商 
品 信息 的 目的 。 在 对 商品 分 类 进行 调试 时 ， 出 现 如 图 4.30 和 图 4.31 所 示 的 现象 。 
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血 动 详情 
计算 机 图 书 玩具 类 网 络 戎 戏 服饰 家 居 日 用 电影 埋头 数码 产品 通讯 设备 电脑 软件 


页 县 
-- | 4 Rr 
i | 
WC | 商品 258 :时 BOWE 商品 型号 : invE-99 
BT | $087 商品 简介 : ”页 着 
Gd 上 而 BM: zoo-t-! La] 商品 竺 角 : 珊 吕 
ss) 所 价 : geen 元 会 员 价 ; Sm 元 。。 折扣 ;96% 
Se eS 有 品名 称 : 请 晤 显示 者 CE FS 
SRYE ELI Waa: Lo 商品 型号: LE- 
本 出 城 从 即日 起 推出 全 可 笑 鉴 商品 简介 : 。 改 瑟 
所 有 会 页 商品 一 时 T5 折 居 台 上 而 BM: 2mori-1 光量 商 8 等 伯 : 一 各 
。 电子 商务 系统 全 面 推出 病 财 加 商场 价 ; 1499 元 会 员 价 ; 1396 元 新 加 ;94 
Ee (和 
(Mi ne Em: mi 
ME et 2 商品 弄 号: -S001 
三 dt eee 2008-5-5 2 
eA 商场 价 : 259 元 : Tr 
一 生 Lnvesee 购 物 中 心 EG 
ca cj 本 站 共有 讲 交 商品 《 件 每 页 显示 3 件 第 1 页 / 共 2 页 ! 2 


4.30 商品 的 分 类 显示 


图 4.31 在 翻 页 时 检索 不 到 数据 
在 图 4.30 所 示 的 页 面 中 可 以 看 到 ， 该 类 的 商品 总 数 为 4 件 商品 ， 每 页 显示 3 件 商品 ， 因 此 ， 


数码 


产品 类 别 下 的 商品 应 为 两 页 进行 显示 ， 而 在 单 击 “2” 超 链接 进行 分 页 时 ， 却 弹出 如 图 4.31 所 示 的 “本 


站 暂 无 该 类 产品 ! ”错误 提示 。 按 逻辑 推理 ， 第 2 页 应 该 有 一 件 商品 ， 但 未 显示 。 

经 过 分 析 ， 在 进行 数据 查询 时 ， 是 按 商 品 的 分 类 进行 查询 的 ， 代 码 如 下 : 

$sqll=mysql_query("select * from tb_shangpin where typeid=".$id." order by addtime desc limit 
".($page-1)*$pagesize.".$pagesize ".$conn): 

由 于 $id 的 值 没有 获取 到 ， 所 以 在 翻 页 时 ， 系 统 检索 不 到 商品 信息 。 

解决 方法 : 在 实现 各 个 分 页 数据 传 值 时 ， 将 分 类 商品 的 id 值 “id=<?php echo $id;?>” 也 一 并 
过 去 ， 代 码 如 下 : 


<a href="showfenlei php?id=<?php echo Sid;?>&page=1" title=" 首 页 "> 
<a href="showfenlei php?id=<?php echo Sid;?>&page=<?php echo Spage-1:?>" title=" 前 一 页 "> 


传递 
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<a href="showfenlei php?id=<?php echo Sid;?>&page=<?php echo $page-1:?>" title=" 后 一 页 "> 
<a href="showfenlei. php?id=<?php echo Sid;?>&page=<?php echo $pagecount:?>" title=" 尾 页 "> 


这 样 在 单 击 分 页 的 各 个 链接 后 ， 即 可 检索 到 分 类 商品 $id 的 值 ， 从 而 实现 了 商品 分 页 显示 功能 。 
[| 说明: 商品 分 页 时 显示 的 分 页 文字 标识 ( 如 后 一 页 “ 修 ” ) 的 超 链接 ， 也 要 将 分 类 商品 的 id 值 
“id=<?php echo $id;?>” 也 一 并 传递 过 去 。 部 分 代码 如 下 : 
<a href="showfenlei.php?id=<?php echo Sid;?>&:page=<?php echo $page-1:?>" title=" 后 一 页 "><font 


face="webdings"> 8 </font></a> <!-- 后 一 页 “ 洲 ”文字 标 识 超 链 接 --> 
有 <!-- 省 略 其 他 3 个 文字 标识 超 链接 的 代码 部 分 --> 


4.8 购物 车 模块 设计 


4.8.1 网 站 购物 车 概述 


购物 车 在 电子 商务 平台 里 作为 前 台 用 户 端 程序 中 非常 关键 的 一 个 功能 模块 ， 帮 助 用 户 完成 商品 的 
选 购 ， 并 把 商品 交 给 服务 台 进 行 结算 。 购 物 车 的 管理 框架 如 图 4.32 所 示 。 


购物 车 列表 


! ! ! ! 
| ww | | | sswwma Ed | 清空 则 物 车 | | [ms 


图 4.32 购物 车 管理 框架 图 
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4.8.2 网 站 购物 车 技术 分 析 


在 电子 商务 平台 开发 过 程 中 ， 相 对 较 困 难 而 且 重 点 的 部 分 是 购物 车 的 实现 过 程 。 购 物 车 的 作用 是 
用 来 临时 储存 用 户 的 购物 信息 ， 用 户 可 以 执行 修改 购物 车 中 的 商品 数量 、 移 除 购 物 车 中 的 某 件 商品 、 
清空 购物 车 等 操作 。 开 发 一 个 购物 车 的 方法 有 两 种 : 一 种 是 将 购物 信息 存储 到 数据 表 中 ;， 另 一 种 是 将 
购物 信息 存储 到 Session 变量 中 。 

如 果 在 后 台数 据 库 中 单独 开辟 一 个 数据 表 来 存储 购物 车 中 的 内 容 ， 也 能 够 实现 购物 车 的 制作 ， 但 
这 会 大 大 浪费 数据 库 服 务 器 的 硬盘 空间 ， 并 且 这 些 信息 对 用 户 和 管理 者 来 说 都 是 没有 价值 的 ， 毕 况 购 
物 车 中 的 商品 不 是 用 户 确定 要 购买 的 商品 ， 所 以 购物 车 应 该 是 临时 存储 用 户 打 算 购买 商品 的 地 方 ， 并 
且 购 物 车 中 的 商品 也 应 该 随 着 用 户 的 退出 而 清空 。 

考虑 到 以 上 因素 笔者 联想 到 Session， 购 物 车 的 实现 和 数据 存储 可 以 依赖 Session 实现 。 为 了 便于 
理解 ， 可 以 把 购物 车 和 超市 联想 起 来 。 首 先 应 该 为 每 位 光临 的 顾客 分 配 一 个 购物 车 ， 当 用 户 成 功 登录 
后 ， 为 用 户 分 配 一 个 $producelist 变量 和 一 个 $quatity 变量 ， 分 别 用 来 存储 用 户 放 入 购物 车 中 商品 的 id 
和 该 商品 对 应 的 数量 ， 并 且 $producelist 的 初始 值 为 室 ， 而 变量 $quatity 的 初始 值 为 1。 

用 户 如 果 选 择 某 件 商品 并 打算 将 其 放 入 购物 车 中 ， 只 要 用 该 Sproducelist 变量 原来 的 值 加 上 新 放 入 
购物 车 中 的 id 值 再 加 上 字符 @, 同时 变量 $quatity 应 在 原来 的 基础 上 加 1 再 加 @, 这 样 就 能 实现 将 商品 
添加 到 购物 车 并 使 该 商品 初始 数量 为 1。 

如 果 用 户 打 算 修改 某 件 商品 的 数量 只 需 用 explode0 函 数 提取 该 商品 的 id 值 和 该 商品 此 时 的 数量 
值 ， 并 将 该 id 对 应 的 存储 在 变量 $quatity 中 的 该 商品 数量 赋 于 新 值 即 可 。 如 果 用 户 打算 将 购物 车 中 某 
件 商品 移 除 ， 只 需 将 该 商品 对 应 的 id 值 赋 于 空 值 ， 并 将 该 商品 对 应 的 数量 赋 于 空 值 。 清 空 购物 车 只 要 
将 变量 $producelist 和 变量 $quatity 同时 赋 于 空 串 即 可 。 


4.8.3 ”添加 至 购物 车 的 实现 过 程 


国 ”添加 至 购物 车 使 用 的 数据 表 : tb_shangpin 
在 网 站 商品 展示 区 ， 单 击 相 应 商品 中 的 “购买 ”按钮 (或 在 商品 详细 信息 页 面 ， 单 击 “ 放 入 购物 
车 ” 超 链 接 ) ， 即 可 进入 到 “添加 至 购物 车 ”页 面 。 添 加 至 购物 车 页 面 的 运行 结果 如 图 4.33 所 示 。 


图 4.33 ”添加 至 购物 车 页 面 的 运行 结果 
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当 用 户 进入 商城 后 ， 一 旦 选 购 了 商品 ， 系 统 就 会 为 每 一 个 用 户 分 配 一 辆 购物 车 供用 户 使 用 ， 并 为 
每 个 用 户 分 配 两 个 session 变量 $producelist 和 $quatity， 分 别 用 来 存储 用 户 放 入 购物 车 中 的 商品 id 和 这 
些 商品 的 数量 , 当然 一 个 变量 同一 时 刻 只 能 有 一 个 值 , 那么 如 何 将 多 个 id 值 同时 保存 在 一 个 $producelist 
变量 中 呢 ? 首先 将 id 转变 成 字符 型 变量 ， 并 且 这 些 变 量 用 字符 “@” 进 行 连接 ， 比 如 用 户 分 别 将 id 
为 1、3、5 的 商品 放 入 购物 车 中 ， 这 时 session 变量 $producelist 的 值 应 该 为 “1@3@5@”。 

当 用 户 不 断 单 击 商品 旁边 的 “购买 ”按钮 时 ， 系 统 将 会 不 停 地 帮 用 户 把 商品 放 入 到 给 用 户 分 配 的 
购物 车 中 。 对 于 相同 的 商品 ， 系 统 会 提示 用 户 已 经 添加 到 购物 车 中 ， 只 需 在 购物 车 中 修改 对 应 的 商品 
的 数量 即 可 。 添 加 至 购物 车 的 代码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\04\shop\addgouwuche.php 


<?php 

session_ start(); 

include("conn/conn.php"): // 连 接 数 据 库 文件 
if($_SESSION[usemame]=—"") { // 判 断 用 户 是 否 已 经 登录 


echo "<script>alert(' 请 先 登录 后 购物 !):history.backO:</script>": // 如 果 用 户 还 没 登录 ， 则 提示 用 户 先 登录 并 返回 
到 原来 页 面 
exit; // 用 exit 语 句 停止 循环 的 继续 执行 


} 
Sid=strval($_GET[id]): // 获 取 商 品 id 值 
$sql=mysql _ query("select * from tb_shangpin Where id=".$id."",$conn); 
$info=mysql_fetch_array($sqD): 
if($info[shuliang]<=0) { // 如 果 商 品 数量 小 于 0， 则 提示 用 户 商 品 已 售 完 
echo "<script>alert(' 该 商品 已 经 售 完 !):history.backO:</script>": 
exit; 
} 
@ Sarray=explode("@".$_SESSION[producelist]): /将 session 变 量 $producelist 中 的 内 容 用 字符 “@” 
进行 分 隔 ， 并 将 结果 保存 在 数组 Sarray 中 
四 forGSi=0:$i<count($array)-1:Si++){ 
这 $aray[$i 一 Sid) { /如 果 $array 数 组 中 存在 与 Sid 相 等 的 元 素 ， 说 明 该 id 所 对 应 的 商品 已 经 在 购物 车 中 
echo "<script>alert(' 该 商品 已 经 在 您 的 购物 车 中 !");:history.backO:</script>"; 
exit; 


} 


} 
@ $_SESSION[producelist]=$_SESSION[producelist] $id."@": 
0 $_SESSION[quatity]=$_SESSION[quatity]."1@": 
header("location:gouwul .php"): /| 添加 成 功 后 ， 重 新 定位 到 gouwul.php 页 面 显 示 购 物 车 中 的 内 容 
?> 


的 代码 贴 二 

@ explode0 函 数 : 以 字符 串 $_ SESSION[producelist] 中 的 子囊 @ 作 为 分 割 符 将 字符 串 分 审 开 来 ， 分 割 后 的 一 个 或 多 
个 子囊 以 数组 的 形式 返回 。 

@ countO 函 数 : 用 来 计算 数组 $array 中 元 素 的 个 数 ， 如 果 变 量 为 空 ， 则 返回 0; 变量 如 果 是 数组 ， 则 返回 数组 元 素 
的 个 数 ， 如 果 是 普通 型 变量 ， 则 返回 1。 

四 $ SESSION[producelist]: 用 来 存储 用 户 放 入 购物 车 中 的 商品 id， 并 用 “(@” 进 行 分 割 。 

@ $_ SESSION[quatity]: 用 来 存储 用 户 放 入 购物 车 中 的 商品 数量 ， 并 用 “(@” 进 行 分 割 ， 默 认 数 量 设置 为 1。 
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[ED 说 明 : 上 面 说 到 的 id 指 的 是 tb_shangpin 表 中 的 过 字段 。 
4.8.4 查看 购物 车 的 实现 过 程 


国 ”查看 购物 车 使 用 的 数据 表 : tb_shangpin 

在 购物 的 过 程 中 ， 当 用 户 购买 完 商 品 或 单 击 “ 我 的 购物 车 ” 超 链 接 后 ， 即 可 在 购物 车 列表 页 面 查 
看 当前 用 户 的 所 购 商 品 的 详细 情况 ， 在 该 页 面 用 户 可 以 对 选 购 的 商品 进行 移 除 、 数 量 更 新 、 结 账 或 者 
清空 购物 车 等 操作 。 查 看 购物 车 页 面 的 运行 结果 如 图 4.34 所 示 。 


[ ] 找 加 密码 
LE 要 商品 光量 者 收 良 各 沽 2 于 eB 计 :3167 


4.34 查看 购物 车 页 面 的 运行 结果 


在 查看 购物 车 页 面 中 ， 将 Sproducelist 用 @ 进 行 分 割 从 而 将 购物 车 中 现 有 商品 id 的 值 存放 到 数组 
$arraygwuc 中 ， 将 session 变量 $quatity 中 的 内 容 用 字符 @ 进 行 分 割 ， 并 将 结果 保存 在 数组 $arrayquatity 
中 ， 然 后 应 用 for 循环 语句 输出 购物 车 中 商品 ， 代 码 如 下 : 


例 程 10 ”代码 位 置 : 光盘 \TMW4\ shop\gouwul.php 


<?php 
session_start(); // 初 始 化 session 变 量 
@ ifs SESSION[username]—""){ /如 果 用 户 名 为 室 ， 则 提示 用 户 先 登录 
echo "<script>alert(' 请 先 登 录 ， 后 购物 !):history.backO:</script>"; 
exit: // 如 果 用 户 没 登录 则 停止 程序 继续 执行 
} 
> 


<table width="500" border="0" align="center" cellpadding="0" cellspacing="1"> 
<form name="form1" method="post" action="gouwul.php"> 
<2php 
session register("total"): /注册 session 变 量 $total 用 来 保存 所 有 商品 价格 总 和 
让 判断 用 GET 方 法 提交 的 gk 的 值 为 yes， 则 将 producelist 和 Squatity 的 值 设 为 空 串 ， 从 而 实现 清空 购物 车 的 目的 */ 
if($_GET[qk}—"yes"){ 
$_SESSION[producelist]j=": 
$_SESSION[quatity]="": 


让 六 六 下 六 六 本 六 六 本 六 六 本 本 六 本 六 亲本 六 六 亲 站 六 本本 六 站 本 六 亲本 六 亲本 冰冰 本 六 亲 让 六 亲 让 本 半 亲 站 六 亲 站 六 本 站 六 亲本 六 六 六 六 让 亲本 六 站 亲 订 冰 bd 
/将 $producelist 用 @ 进 行 分 割 从 而 将 购物 车 中 现 有 商品 和 的 值 存放 到 数组 Sarraygwuc 中 


S$arraygwc=explode("@".$_ SESSION[producelist): 
$s=0:; // 用 $s 保存 购物 车 中 商品 id 的 总 和 
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for(Si=0:$i<count($arraygwc):Si++H){ 


© $s+=-intval($arraygwc[Si]): 
上 
这 $s 一 0){ /如 果 $s 的 值 为 室 ， 则 说 明 购 物 车 中 无 商品 
echo "<tr>"; 
echo" <td height-'25' colspan='6'bgcolor=#FFFFFF' align='center> 您 的 购物 车 为 空 !</td>": 
echo"</tr>": 
由 
else{ // 否 则 ， 显 示 购 物 车 中 的 所 有 商品 信息 
?> 
<t> 
<td width="125" height="25" bgcolor="#FFFFFF"><div align="center"> 商 品名 称 </div></td> 
// 显 示 购 物 标题 名 称 HTML 标 记 部 分 略 
</t> 
<?php 
Stotal=0; 


// 将 session 变 量 $producelist 中 的 内 容 用 字符 @ 进 行 分 割 ， 并 将 结果 保存 在 数组 Sarray 中 
Sarray=explode("(@",$ SESSION[producelist]); 
/将 session 变 量 $quatity 中 的 内 容 用 字符 @ 进 行 分 割 ， 并 将 结果 保存 在 数组 Sarrayquatity 中 
Sarrayquatity=explode("@®",S SESSION[quatity]); 
$_SESSION[quatity]=implode("(@",$arrayquatity): 
for($i=0:$i<count($array)-1:$iH+){ 
Sid=S$array[$i]: 
Snum=$arrayquatity[$i]: 
/* 中 志 坟 沁 直 兴 相 相 相 如果 Sid 不 为 空 ， 则 从 商品 信息 表 中 获取 指定 商品 id 的 信息 **** 寺 二/ 
if($id!=""){ 
$sql=mysql_query("select * from tb_shangpin where id=".$id."",$conn); 
Sinfo=mysql] fetch_array($sq]l); 


Stotall=$num*$info[huiyuanjia]: /商品 金额 = 选 购 数量 X 会 员 价 
S$total+=$totall; /累计 购物 车 中 的 商品 金额 
$_SESSION["total"]=S$total: // 购 物 车 中 的 商品 的 累计 金额 
Pd 率 束 束 束 事 束 束 事 束 束 事 束 素来 事 束 束 事 束 束 事 束 束 事 事 事 事 事 束 事 事 事 事 事 事 束 可 事 事 束 事 束 素 束 束 束 事 束 事 事 束 事 束 束 事 束 事 事 素 束 束 束 束 束 束 来 束 率 束 束 束 事 素 */ 
?> 
ea 显示 购物 商品 的 相关 信息 一 = 
<tr> 


<td height="25"><?php echo $info[mingcheng]:?> </td> 

<td height="25"><input type="text" name="<?php echo $info[id]:?>" size="2" class="inputcss" value=<?php echo 
$num:?>> </td> 

<td height="25" ><?php echo $info[shichangjia]:?> 元 </td> 

<td height="25"><?php echo $info[huiyuanjia]:?> 元 </td> 
@ <td height="25"><?php echo @(ceil((Sinfo[huiyuanjia]/Sinfo[shichangjia])*100))."%":?> </td> 

<td height="25"><?php echo $info[huiyuanjia]j*$num." 元 ":?></td> 

<td height="25"><a href="removegwc.php?id=<?php echo $info[id]?>"> 移 除 </a></td> 

</t> 
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<tr> 
<td width="125"> 总 计 : <?php echo Stotal: ?> </td> 


?> 
</table> 


< 人) 代码 贴 二 


限 ， 


@ $5_SESSION[username]: 当 用 户 登 录 网 站 后 ， 系 统 应 用 Session 变量 记录 用 户 名 。 只 有 会 员 可 以 有 操作 网 站 的 权 
如 果 用 户 名 为 空 ， 则 提示 用 户 先 登录 。 

@@ intval: 对 购物 车 中 的 商品 这 值 取 整 。 

@ ceil: 对 会 员 价 除 以 市 场 价 后 的 小 数 部 分 进 一 取 整 ， 如 相 除 后 的 结果 为 43， 应 用 ceil(4.3) 后 的 结果 为 5。 

@ Stotal: 购物 车 中 的 所 有 商品 的 的 累计 金额 。 


4.8.5 ”从 购物 车 中 移 去 指定 商品 的 实现 过 程 


国 。 从 购物 车 中 移 去 指定 商品 使 用 的 数据 表 : tb_shangpin 
添加 “ 移 除 ” 文 字 超 链接 ， 删 除 id 指定 的 商品 信息 。 
<a href="removegwc.php?id=<?php echo $info[id]?>"> 移 除 </a> 


购物 车 的 作用 是 用 来 临时 储存 用 户 的 购物 信息 ， 因 此 用 户 可 以 随时 对 购物 车 中 的 商品 进行 移 除 。 


在 查看 购物 车 页 面 中 ， 单 击 对 应 商品 后 的 “ 移 除 ” 超 链 接 ， 即 可 将 商品 信息 从 购物 车 中 删除 。 


该 功能 实现 的 基本 思想 是 : 首先 用 函数 explode0 将 session 变量 $producelist 以 @ 进 行 分 割 ， 并 把 分 


割 出 的 子 串 存 放 到 数组 中 ， 然 后 将 用 户 打算 移 去 商品 对 应 的 数组 元 素 赋 于 空 值 ， 最 后 将 数组 元 素 重新 
组 合成 新 串 。 其 代码 如 下 : 


例 程 11 代码 位 置 : 光盘 \TM\04\shopremovegwc.php 


<?php 
session_start(): // 初 如 化 Session 变 量 
$id=$_GET[id]: // 获 取 用 户 打算 移 去 商品 的 id 
Sarraysp=explode("(@".$_SESSION[producelist]): // 将 购物 车 中 的 商品 id 存 储 到 数组 $arraysp 中 
Sarraysl=explode("@".$_SESSION[quatity]): // 将 购物 车 中 的 商品 的 数量 存放 到 数组 $arraysl 中 
for($i=0:$i<count($arraysp):$it+) { 
这 $arraysp[$i] 一 Sid) { // 通 过 循环 寻找 与 id 值 相等 的 数组 元 素 
Sarraysp[Sil=""; 
Sarraysl[Si]=""; // 将 与 id 值 相等 的 数组 元 素 赋 于 空 值 


} 


} 
$_SESSION[producelist]j=implode("@",$arraysp): 
$_SESSION[quatity]=implode("@".Sarraysl): /应 用 implode0 函 数 将 数组 元 素 重新 组 合成 新 串 
header("location:gouwul php"): // 重 新 定位 到 gouwul.php 显 示 购物 车 中 的 商品 信息 
?> 
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发 技巧 : 如 果 程 序 中 应 用 到 了 session 变量 ， 那 么 就 需要 对 session 变量 进行 初始 化 ， 在 程序 的 首 行 
添加 “session_start0;” 语 句 ， 如 果 没 有 对 session 变量 进行 初始 化 或 者 位 置 不 是 在 首 行 ， 那 
么 程序 在 运行 时 会 弹出 相应 的 错误 信息 。 


4.8.6 修改 商品 购买 数量 的 实现 过 程 


国 ”修改 商品 购买 数量 使 用 的 数据 表 : tb shangpin 

购物 车 中 的 商品 默认 购买 数量 是 1 件 ， 如 果 用 户 打算 购买 多 件 相同 的 商品 就 需要 利用 修改 文本 杠 
中 的 商品 数量 来 实现 。 修 改 商 品 数量 与 从 购物 车 中 移 去 指定 商品 的 原理 类 似 ， 只 不 过 从 购物 车 中 移 去 
某 件 商品 是 将 该 商品 对 应 的 数组 元 素 赋 于 空 值 ， 而 修改 商品 购买 数量 是 将 购物 车 中 某 件 商品 对 应 的 数 
组 元 素 赋 于 新 值 。 其 代码 如 下 : 


例 程 12 ”代码 位 置 : 光盘 \TM\04\ shop\gouwul.php 
@ while(list(Sname,Svalue)=each($_ POST)) { /提取 表单 中 的 商品 id 和 新 数量 
for($i=0;$i<count($array)-1:$i++){ 
if(Sarray[$i])=—=S$name) { 
© Sarrayquatity[Si]=Svalue; // 获 取 购 物 车 中 每 种 商品 的 数量 ， 并 将 数量 保存 到 $arrayquatity 数 组 中 
} 
} 
} 


< 代码 贴 二 

@ list($name,$value)=each($_POST): 查看 购物 车 页 面 中 ， 输 入 商品 数量 的 文本 框 是 用 商品 id 来 命名 的 ， 所 以 此 处 
将 提交 表单 元 素 的 名 称 和 值 依次 存储 到 对 应 的 变量 Sname 和 Svalue 中 。 

@ $arrayquatity[$i]=$value: 获取 购物 车 中 每 种 商品 的 新 数量 ， 并 将 新 数量 值 保存 到 Sarrayquatity 数组 中 。 


4.8.7 ”清空 购物 车 的 实现 过 程 


在 查看 购物 车 页 面 添加 “清空 购物 车 ”文字 超 链接 ， 代 码 如 下 : 

<a href="gouwul.php?qk=yes"> 清 空 购物 车 </a> 

当 用 户 想 重新 选 购 商品 时 ， 需 要 清空 购物 车 中 所 有 商品 ， 将 session 变量 $producelist 和 $quatity 的 
值 都 赋 于 空 串 。 其 代码 如 下 : 

例 程 13 ”代码 位 置 : 光盘 \TM\04\ shop\gouwul.php 


if($_GET[qk]—"yes"){ // 判 断 用 户 是 否 单 击 “ 清 空 购物 车 ” 超 链接 
$_SESSION[producelist]=""; // 清 空 购物 车 中 商品 id 
$_SESSION[quatity]="": // 清 空 购物 车 中 商品 数量 


» 
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4.8.8 ”收银 台 结 账 的 实现 过 程 


国 ”收银 台 结账 使 用 的 数据 表 : tb_shangpin、tb user、tb_dingdan 
当 用 户 选 购 完 商 品 后 ， 在 购物 车 列表 页 面 单 击 “ 去 收银 台 ” 超 链接 ， 即 可 进入 到 “收银 台 结 账 ” 
页 面 ， 运 行 结果 如 图 4.35 所 示 。 


* 程序 正在 测 式 中 

* 南城 的 价格 已 经 打 啊 

“本 商城 从 即日 起 推出 优惠 活 
* 所 有 会 员 商品 一 建 15 折 忧 圳 
* 电子 商务 系统 全 面 推出 天 寿 


图 4.35 ”收银 台 结账 页 面 的 运行 结果 
在 查看 购物 车 页 面 添加 “去 收银 台 ” 超 链接 ， 实 现 所 购买 商品 的 金额 结算 功能 ， 代 码 如 下 : 
<a href="gouwu2.php"> 去 收银 台 </a> 
单 击 “ 去 收银 台 ” 超 链接 ， 跳 转 到 gouwu2.php 页 ， 该 页 面 主要 设计 用 户 需 要 填写 的 订单 结构 。 根 
据 用 户 在 购物 车 页 面 提交 的 商品 信息 ， 为 用 户 提供 填写 订单 的 平台 ， 然 后 将 用 户 选 购 的 商品 信息 〈 包 


括 商品 名 称 、 商 品 数量 等 ) 以 及 订单 信息 存储 在 数据 库 中 。 
收银 台 页 面 涉及 到 的 HIML 表单 的 重要 元 素 如 表 4.1 所 示 。 


表 4.1 收银 台 页 面 涉及 到 的 HTML 表单 的 重要 元 素 


重要 属性 
method="post" action="savedd.php" onSubmit="return 
chkinput(this)” 

<option selected value=" 普 通 平邮 "> 普通 平邮 </option> 
<option value=" 特 快 专递 "> 特快 专递 </option> 

shff select <option value=" 送 货 上 门 "> 送 货 上 门 </option> 

<option value=" 个 人 送 货 "> 个 人 送 货 </option> 

<option value="E-mail">E-mail</option> 


forml 
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续 表 
名 称 | 元 素 重要 属性 含义 
<option selected value=" 建 设 银行 汇款 "> 建设 银行 汇款 </option> 
2 <option value=" 交 通 银 行 汇款 "> 交通 银行 汇款 </option> 货款 的 支付 方式 


<option value=" 邮 局 汇款 "> 邮局 汇款 </option> 
<option value=" 网 上 支付 "> 网 上 支付 </option> 


sibmio | suomit | ass | "失实 i" 摘 角 


用 户 确定 要 购买 购物 车 中 的 所 有 商品 之 后 ， 就 需要 到 收银 台 页 面 填写 收 货 人 信息 ， 系 统管 理 人 员 
将 通过 该 信息 确定 收 货 人 地 址 、 商 品名 称 及 数量 等 信息 ， 系 统 同 时 会 根据 这 些 信 息 给 出 订单 。 具 体 实 
现代 码 如 下 : 

例 程 14 ”代码 位 置 : 光盘 \TM\04\shop\savedd.php 


<?php 
Session _start(): /初始 化 session 变 量 
include("conn/conn.php"); // 连 接 数 据 库 文件 
$sql=mysql_query("select * from tb_user Where name=".$_SESSION[username]."",$conn); 
Sinfo=mysql_fetch_array($sql): /检索 用 户 数据 表 中 的 信息 
Sdingdanhao=date("YmjHis").$info[id]: // 订 单 号 = 当前 日 期 时 间 + 用 户 的 id 号 
$spc=$_SESSION[producelist]: // 将 用 户 购 买 的 商品 名 称 串 赋 给 变量 $spc 
$slc= $_SESSION[quatity]: // 将 用 户 购 买 的 商品 数量 串 赋 给 变量 $slc 
$shouhuoren=$_POST[name2]; // 获 取 收 货 人 姓名 
S$sex=$ POST[sex]: /获取 收 货 人 性 别 
S$dizhi=$ POST[dz]: /获取 收 货 人 地 址 
Syoubian=$_POST[yb]: // 获 取 收 货 人 邮编 
Stel=$_POSTT[tel]: // 获 取 收 货 人 电话 
Semail=$_POST[email]: /获取 收 货 人 E-mail 地 址 
$shff=$_POST[shff]: // 获 取 送 货 方式 
$zfff-$_POST[zfff]: /获取 支付 方式 
这 trim($_ POST[Iy]) 一 ""){ /如 果 用 户 留言 为 空 
Sleaveword=""; // 则 将 Sleaveword 变 量 设 为 空 
else{ // 否 则 获取 用 户 的 留言 信息 
Sleaveword=$_POST[Iy]: 

} 

S$xiadanren=$ SESSION[username]: // 获 取 下 单 人 名 称 ， 即 当前 用 户 

Stime=date("Y-m-j H:is // 获 取 系统 当前 时 间 

S$zt=" 未 作 任何 处 理 "; 

Stotal=$_SESSION[total]: // 获 取 购 物 车 内 所 有 商品 的 累计 金额 


Imysql_ query("insert into 
tb dingdan(dingdanhao,spc.slc,shouhuoren.sex.dizhi,youbian.tel.email.shff.zfff.leaveword.time.xiadanren.zt.total) values 
(‘$dingdanhao'.'$spe','$slc'.'$shouhuoren'.'$sex','$dizhi','$youbian'.'$tel'.'Semail'.'$shf?.'$zfff.'$leaveword'.'$time'.'$xiadanren','$z 


t','Stotal)".Sconn): // 将 信息 添加 到 tb_dingdan 表 
header("location:gouwu?2.php?dingdanhao=Sdingdanhao"): // 重 新 定位 到 收银 台 
?> 
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总 技巧: 上述 代码 中 mysql query0 浮 数 的 执行 结果 并 没有 具体 的 赋 给 菜 个 变量 ， 这 是 因为 当 用 该 函 
数 执行 insert、delete 等 SQL 语句 时 并 不 需要 回 显 结果 ， 而 执行 select 语 句 时 一 般 都 需要 将 
结果 显示 在 前 台 页 面 中 ， 所 以 经 常会 将 该 函数 的 执行 结果 赋 给 某 一 个 变量 ， 以 后 用 诸如 
mysql fetch_arrayO) 函 数 提 取 数 据 库 中 内 容 时 会 用 到 该 变量 。 


4.8.9 生成 商品 订单 的 实现 过 程 

国 ”生成 商品 订单 使 用 的 数据 表 : tb shangpin、tb_dingdan 

在 收银 台 结 账 页 面 ， 单 击 “ 提 交 订单 ” 按 钮 ， 即 可 进入 “生产 商品 订单 ”页 面 。 该 页 面 的 运行 结 
果 如 图 4.36 所 示 。 


巷 喜 纯净 水 ， 悠 已 成 功 的 提交 了 此 订单 | 详细 信息 如 下 : 
订单 号 : 2007112214363739 


市 场 价 会 员 价 
2699 2599 


纯净 水 


Ea 
: 吉林 省 长 春 市 河东 路 14* 号 
130000 电 话 ; 0431-849TSeer 


riraanpsgzis on 
: 。 送 货 上 门 支付 方式 : 网 上 支付 
读 低 在 一 周 内 按 修 的 支付 方式 进行 汇款 ,汇款 时 注 明 作 的 订单 号! 汇款 后 请 及 时 通知 我 们 
关 疝 窗口 自 娃 时间 : 2007-11-22 14:38:37 


4.36 生成 商品 订单 页 面 的 运行 结果 


提交 表单 信息 到 数据 处 理 页 ， 将 商品 及 用 户 信息 存储 到 订单 信息 表 中 ， 添 加 成 功 后 将 生成 的 订单 
号 $dingdanhao 传递 到 收银 台 页 ， 代 码 如 下 : 


例 程 15 ”代码 位 置 : 光盘 \TM\04\shop\gouwu2.php 


<?php 

if($_GET[dingdanhao]!=") { // 如 果 订 单 号 不 为 空 
$dd=$_GET[dingdanhao]: // 获 取 订 单 号 
session start(): // 初 始 化 session 变 量 


/以 子囊 @ 作 为 分 割 符 将 字符 串 分 割 开 来 ， 分 割 后 的 一 个 或 多 个 子 串 以 数组 的 形式 返回 
Sarray=explode("@".$_SESSION[producelist]): 


Ssum=count($array)*20+260: // 计 算 窗 体 的 显示 高 度 
echo" <script language='javascript>": //JavaScript 脚 本 标识 语句 开始 
echo" 
Window.open('showdd.php?dd='+".$dd.",'newframe','top=150.,left=200,width=600.height=".$sum.".menubar=no.toolbar=no.lo 
cation=no.scrollbars=no.status=no )": // 在 新 窗口 showdd.php 页 中 输出 订单 信息 
echo "</script>": //JavaScript 脚 本 标识 语句 结束 
} 


?> 


应 用 Window 对 象 的 open 方法 打开 一 个 新 窗口 showdd.php， 用 于 显示 商品 订单 的 详细 信息 ,将 商 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


品 串 以 @ 子 串 进行 分 割 ， 返 回 数组 Sarraysp， 将 数量 串 以 @ 子 串 进行 分 割 ， 返 回 数组 $arraysl， 然 后 应 用 
for 循环 语句 循环 输出 订单 中 的 商品 信息 ， 代 码 如 下 : 
例 程 16 ”代码 位 置 : 光盘 \TM\04\shop\showdd.php 


<?php 
session start(; // 初 始 化 session 变 量 
include("conn/conn php"): // 连 接 数 据 库 文件 
S$dingdanhao=$_GET[dd]: // 获 取 订 单 号 
$sql=mysql query("select * 人 om tb_dingdan where dingdanhao="".$dingdanhao.”"", $conn); 
Sinfo=mysql fetch array($sql): /查询 订单 号 所 对 应 的 订单 详细 信息 
$spc=$info[spc]: /获取 商品 串 
S$slc=$info[slc]: /获取 数量 串 
Sarraysp=explode("(@",$spc): /将 商品 串 以 @ 子 串 进行 分 割 ， 返 回 数组 $arraysp 
Sarraysl=explode("(@".$slc): /将 数量 串 以 @ 子 串 进行 分 割 ， 返 回 数组 $arraysl 
?> 


/省 略 部 分 HTML 代 码 
<td ee "20" bgcolor="#FFEDBF"><div align="center" class="style7"> 恭 喜 <?php echo $_SESSION[username]:?>， 
您 已 成 功 的 提交 了 此 订单 ! 详 细 信 息 如 下 :</div></td> 
<td height="20" bgcolor="#FFFFFF"><div align="left"><span class="style5">&nbsp: 订 单 号 ，</span><?php echo 


$dingdanhao:?></div></td> 

<?php 

Stotal=0; 

for($i=0;$i<count($arraysp)-1:$i+H+){ // 用 for 循 环 语句 循环 输出 订单 中 的 商品 信息 

if(S$arraysp[Si]!=""){ // 如 果 商 品 不 为 空 

$sqll=mysql_query("select * from tb_shangpin where id=".$arraysp[$i]."",$conn); 
S$infol=mysql_fetch_array($sql]): /检索 指定 商品 的 详细 信息 
Stotal=$total+=$arraysl[$i]*$infol[huiyuanjia]:; // 获 取 订 单 中 的 商品 总 计 费 用 

人 = 

下 -输出 订单 商品 的 详细 信息 -一 -一 -一 一 一- > 


<tr bgcolor="#FFFFFF"> 
<td height="20"><div align="center"><?php echo $infol[mingcheng]:?></div></td> 
<td height="20"><div align="center"><?php echo $infol[shichangjia]:?></div></td> 
<td height="20"><div align="center"><?php echo $infol[huiyuanjia]:?></div></td> 
<td height="20"><div align="center"><?php echo $arraysl[$i]:?></div></td> 
<td height="20"><div align="center"><?php echo $arraysl[$i]*$infol[huiyuanjia]:?></div></td> 


</t> 
<2?php 
} 
} 
> 
<tr bgcolor="#FFFFFF"> 
<td height="20" colspan="5"> 
<div align="right"><span class="style5"> 总 计 费 用 :</span><?php echo S$total:?> </div></td> 
</t> 
二 /订单 详细 信息 展示 部 分 代码 略 
去 过 过 二 = a a es a 
S$_SESSION[producelist]="": // 清 空 购物 车 中 的 商品 串 
$_SESSION[quatity]="": // 清 空 购物 车 中 的 数量 串 


?> 
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4.8.10 ”单元 测试 


在 开发 完 购物 车 模块 后 ， 为 了 保证 程序 正常 运行 ， 一 定 要 对 模块 进行 单元 测试 。 单 元 测试 在 程序 
开发 中 非常 重要 ,只 有 通过 单元 测试 才能 发 现 模块 中 的 不 足 之 处 ， 从 而 及 时 地 弥补 程序 中 出 现 的 错误 。 
在 开发 购物 车 模块 时 需 注意 : 在 购物 车 列表 页 面 更 改 商 品 数 据 时 ， 小 计 金 额 与 统计 金额 没有 按 更 改 的 
数量 进行 计算 ， 如 图 4.37 所 示 。 


更 改 商品 的 购买 数量 ， 商 品 的 小 计 与 总 计 金 额 未 改变 


4.37 更 新 商品 数量 时 统计 数据 不 准确 


原因 : 购物 车 中 用 于 输入 商品 数量 的 文本 框 是 用 商品 id 来 命名 的 , 所 以 通过 语句 list($name,$value) 
=each($_POST) 赋 值 就 使 Sname 为 购物 车 中 某 件 商品 的 id， 而 $value 为 该 商品 对 应 的 数量 。 

经 过 分 析 ， 从 数组 $array 中 取 值 时 ， 应 该 从 第 1 个 下 标 元 素 开始 取 值 ， 即 $array[0]， 而 $i 的 初始 值 
是 1， 说 明 数 组 是 从 第 2 个 元 素 开始 取 值 的 ， 所 以 在 更 改 商 品 数量 时 ， 商 品 id 与 数量 对 应 不 上 ， 而 小 
计 及 总 计 金 额 都 是 通过 数量 值 进行 计算 的 ， 由 于 数量 没有 获取 到 ， 所 以 程序 中 所 购 商 品 的 数据 统计 不 
准确 。 程 序 中 错误 的 代码 如 下 : 


<?php 
Sarray=explode("@".$_SESSION[producelist]): // 应 用 数组 存储 商品 串 
Sarrayquatity=explode("(@".$_SESSION[quatity]): // 应 用 数组 存储 商品 数量 串 
while(list($name,$value)=each($_POST)){ // 提 取 表 单 中 的 商品 id 和 新 数量 
for(Si=1;Si<=count(Sarray);SitH){ /应 用 for 循 环 语句 获取 商品 这 所 对 应 的 新 数量 
if(($array[$i])—$name){ // 如 果 商 品 的 id 与 更 新 的 数量 相等 
// 获 取 购 物 车 中 每 种 商品 的 数量 ， 并 将 数量 保存 到 $arrayquatity 数 组 中 
Sarrayquatity[$i]=$value: 
} 
} 
?> 
解决 方法 : 


应 用 for 循环 语句 将 数组 中 的 元 素 值 赋 于 新 的 商品 数量 ， 从 数组 $array 中 取 值 时 ， 应 该 从 第 1 个 下 
标 元 素 〈 即 数组 的 第 0 个 元 素 ) 开始 取 值 到 数组 的 最 大 下 标 -1 结束 ， 即 可 正确 统计 商品 的 小 计 与 总 计 


金额 。 更 改 后 的 代码 如 下 : 
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<7 


Sarray=explode("(@".$_SESSION[producelist]): // 应 用 数组 存储 商品 串 
Sarrayquatity=explode("@".$_SESSION[quatity]): // 应 用 数组 存储 商品 数量 串 
while(list($name, $value)=each($ POST)){ // 提 取 表 单 中 的 商品 id 和 新 数量 
for($i=0;Si<=count(Sarray) 1;Si++){ // 应 用 for 循 环 语句 获取 商品 id 所 对 应 的 新 数量 
if(($array[$i])—$name){ // 如 果 商 品 的 id 与 更 新 的 数量 相等 


/获取 购物 车 中 每 种 商品 的 数量 ， 并 将 数量 保存 到 Sarrayquatity 数 组 中 
Sarrayquatity[$i]=$value: 
} 
} 


?> 


4.9.1 后 台 首 页 概述 


后 台 首页 承载 并 显示 网 站 后 台所 包含 的 模块 ， 使 网 站 管理 员 能 够 清楚 其 管理 权限 。 根 据 需 求 分 析 ， 
确定 电子 商务 平台 网 的 后 台 系 统 包括 以 下 功能 模块 : 
商品 信息 管理 模块 : 主要 包括 商品 信息 的 添加 、 修 改 、 删 除 和 商品 类 别 的 添加 。 
用 户 信息 管理 模块 : 主要 包括 查询 和 显示 用 户 注册 信息 、 冻 结 用 户 、 用 户 留言 管理 、 更 改 管 
理 员 密码 。 
订单 信息 管理 模块 : 主要 包括 查看 所 有 用 户 提 交 的 订单 信息 ， 并 根据 执行 阶段 对 订单 进行 标 
记 处 理 、 删 除 订单 和 查询 订单 。 
公告 信息 管理 模块 :主要 包括 站 内 公告 信息 的 添加 、 修 改 、 删 除 和 用 户 评论 信息 的 查看 和 删除 。 
电子 商务 平台 网 的 后 台 首页 分 为 网 站 Banner 浮动 框架 、 导 航 浮动 框架 、 内 容 浮 动 框架 3 个 部 分 。 
下 面 看 一 下 本 案例 中 提供 的 商城 后 台 首 页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\04\shop\admin\ 
defaultphp， 如 图 4.38 所 示 。 


4.9.2 后台 首 页 技术 分 析 


在 网 站 后 台 管理 系统 的 首页 面 中 使 用 浮动 框架 来 规划 页 面 布局 。 浮 动 框架 的 作用 是 把 浏览 器 窗口 
划分 成 若干 个 区 域 ， 每 个 区 域内 可 以 显示 不 同 的 页 面 ， 并 且 各 个 页 面 之 间 不 会 受到 任何 影响 ， 为 框架 
内 每 个 页 面 命名 ， 作 为 彼此 互动 的 依据 。 

在 后 台 首页 面 中 先 使 用 左右 浮动 框架 进行 页 面 布局 。 这 样 ， 就 可 以 在 页 面 顶 端 设置 网 站 的 Banner， 
在 页 面 左 侧 设置 网 站 的 导航 功能 ， 在 页 面 的 右 侧 设置 后 台 系 统 显示 的 主要 的 信息 内 容 。 

浮动 框架 <IFRAME> 是 一 种 特殊 的 框架 结构 ， 它 可 以 在 浏览 器 窗口 中 嵌 套 另外 的 网 页 文件 。 其 语 
法 格式 如 下 : 

<iframe srtc=" 文 件 ” name=" 框 架 名 称 ”align=" 对 齐 方式 ”width=" 值 ”height=" 值 ”scrolling=" 值 ”frameborder=" 值 " 
marginwidth=" 值 " marginheight=" 值 "> 

</ 达 ame> 
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划 欣 电子 靖 械 sees 网 站 Banner 浮动 框架 

可 本 和 | 全 2+ | 人 于 | 几 37 式 E23 E23 
es Bf | 0 | 天 本 区 | Sa EEC EC 
a | 1 | | EEC EN 
-ME | | | EL 
和 | Sn | | | 
i | | | | 
。 用 户 吕 和 | wm | | Hm EEC | MT 
. | 条 | sm | RE | WA EEC 
导航 浮动 框架 | wis | sm | mici| arm | ERA ERA | MT 
训 | Sm | RC | EE 
2 | 0 | 本 EC 
* MM af | BR | 条 内 容 浮动 框架 EC 
ED] EECETONRECIEE 本 本: | 
EE E23 EE 
ne EE E33 Em 
ee nn | War | mw | 5 | Ea EEC 
ET Ea En 
i | | | | | ER EB 
Er E33 | 
i | Mr | i | SIM | | 时 人 ED | MF 
SE EI 


图 4.38 ”电子 商务 平台 后 台 首 页 
下 面 是 浮动 框架 属性 的 详细 讲解 。 
1. 浮动 框架 的 文件 路 径 属性 SRC 
语法 : 
<iframe src="file name"> 
file_ name: 表示 浮动 框架 文件 的 文件 名 或 者 其 他 超 链 接 的 网 址 。 
2. 浮动 框架 的 名 称 属性 name 
语法 : 
<iframe src="file name" name="frame name"> 
frame_name: 定义 的 浮动 框架 名 称 。 
3， 浮 动 框架 的 对 齐 属性 align 
语法 : 
<iframe src="file name" align="left/center/right"> 
left: 居 左 对 齐 。 
回 “center: 居中 对 齐 。 
right: 居 右 对 齐 。 
4. 浮动 框架 的 宽度 和 高 度 属性 width、height 
语法 : 
<iframe src="file name" width="value" height="value"> 
width: 浮动 框架 的 宽度 。 
height: 浮动 框架 的 高 度 。 
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5. 浮动 框架 滚动 条 显示 属性 scrolling 


<iframe src="file name" scrolling="value"> 

value 有 3 个 取 值 。 

yes: 显示 滚动 条 。 

no: 不 显示 滚动 条 。 

auto: 根据 窗口 内 容 决 定 是 否 有 滚动 条 。 

6. 浮动 框架 边框 属性 frameborder 

语法 : 

<iframe src="file name" frameborder="value"> 

value: 值 为 yes 代表 显示 框架 边框 ， 值 为 no 代表 隐藏 框架 边框 。 
7. 浮动 框架 边缘 的 宽度 和 高 度 属性 marginwidth、marginheight 
语法 : 

<iframe src="file name" marginwidth="value" marginheight="value"> 
marginwidth: 设 定 浮动 框架 左右 边缘 与 边框 的 宽度 。 
marginheight: 设 定 浮动 框架 上 下 边缘 与 边框 的 高 度 。 


加 


办 办 的 


4.9.3 ”后 台 首 页 的 实现 过 程 


网 站 Banner 浮动 框架 的 代码 如 下 : 


例 程 17 ”代码 位 置 ; 光盘 \TMVWO4\shopvadminvdefanltphp 

<IFRAME frameBorder=0 id=top name=top scrolling=no src="top.php" 
style="HEIGHT: 90px: VISIBILITY: inherit WIDTH: 1003px: Z-INDEX: 3"> 

</IFRAME> 


导航 浮动 框架 的 代码 如 下 : 


例 程 18 代码 位 置 : 光盘 \TMVWO4\shopvadminvdefaultphp 
<IFRAME frameBorder=0 id=left name=left src="left.php" 

style="HEIGHT: 100%: VISIBILITY: inherit; WIDTH: 212px: Z-INDEX: 2"> 
</IFRAME> 


内 容 浮 动 框架 的 代码 如 下 : 


例 程 19 代码 位 置 : 光盘 \TMVWO4\shopvadminvdefaultphp 


<IFRAME frameBorder=0 id=main name=main scrolling=yes src="lookdd.php" 
style="HEIGHT: 100%: VISIBILITY: inherit WIDTH: 778px: Z-INDEX: 1"> 
</IFRAME> 
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4.10 客户 订单 信息 管理 模块 设计 


系统 管理 员 是 根据 用 户 订单 来 发 放 货物 的 ， 因 此 客户 信息 管理 模块 在 电子 商务 平台 网 中 也 起 着 非 
常 重要 的 作用 ， 它 直接 影响 着 商家 的 信誉 和 消费 者 的 权益 。 


4.10.1 客户 订单 信息 管理 模块 概述 


客户 订单 信息 管理 模块 作为 后 台 核 心 的 功能 ， 主 要 用 于 执行 订单 已 收 款 、 已 发 货 、 已 收 货 的 过 程 ， 
把 订单 交 给 服务 器 进行 处 理 。 客 户 订单 信息 的 管理 框架 如 图 4.39 所 示 。 


客户 订单 列表 
| |smsmam | ms | | 执行 客户 订单 | | 查找 客户 订单 | 


是 否 删除 订单 
本 < 


加 4.39 ”客户 订单 信息 的 管理 框架 


全 注意 : 为 了 保证 商城 的 安全 ， 即 使 是 管理 员 ， 对 于 订单 的 管理 也 不 能 拥有 过 大 的 权限 ， 所 以 管理 
员 只 可 以 查看 订单 并 选择 订单 的 执行 状态 ， 而 不 允许 对 订单 信息 进行 修改 。 


4.10.2 客户 订单 信息 管理 模块 技术 分 析 


客户 订单 信息 管理 模块 的 核心 在 于 处 理 订 单 及 打印 订单 ， 下 面 对 打 印 客户 订单 进行 技术 剖析 。 本 
模块 中 的 打印 订单 功能 主要 应 用 正 内 置 的 WebBrowser 控件 实现 客户 订单 的 打印 等 功能 ， 该 控件 的 具 
体 参数 如 下 : 

document.all WebBrowser.Execwb(7.1): 表示 打印 预览 。 

document.all WebBrowser Execwb(6.1): 表示 打印 。 

document.all WebBrowser.Execwb(6.6): 表示 直接 打印 。 
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document.all.WebBrowser.Execwb(8,1): 表示 页 面 设置 。 
document.all.WebBrowser.Execwb(1.1): 打开 页 面 。 

document.all WebBrowser Execwb(2.1): 关闭 所 有 打开 的 正 窗口 。 
document.all.WebBrowser.Execwb(4,1): 保存 网 页 。 
document.all.WebBrowser.Execwb(10,1): 查看 页 面 属性 。 
document.all.WebBrowser.Execwb(17,1): 全 选 。 

document.all.WebBrowser.Execwb(22,1): 刷新 。 

document.all.WebBrowser.Execwb(45,1): 关闭 窗 体 无 提示 。 

在 实现 打印 功能 时 ， 首 页 需要 建立 HIML 的 object 标签 ， 调 用 WebBrowser 控件 ， 代 码 如 下 : 


<object id="WebBrowser" classid="ClSID:8856F961-340A-11D0-A96B-00C04Fd705A2" width="0" height="0"> 
</object> 


建立 相关 的 打印 超 链 接 ， 并 调用 WebBrowser 控件 的 相应 参数 实现 打印 预览 、 打 印 、 直 接 打印 、 页 
面 设置 等 常用 功能 。 其 代码 如 下 : 

<a href="#" onClick="document.all.WebBrowser.Execwb(7,1)"> 打 印 预览 </a> 

<a href="#" onClick="document.all.WebBrowser.Execwb(6,1)"> 打 印 </a> 


<a href="#" onClick="document.all.WebBrowser.Execwb(6,6)"> 直 接 打 印 </a> 
<a href="#" onClick="document.all.WebBrowser.Execwb(8,1)"> 页 面 设置 </a> 


办 办 办 办 办 多 凶 办 


4.10.3 查看 客户 订单 信息 的 实现 过 程 


国 ”查看 客户 订单 信息 使 用 的 数据 表 : tb dingdan 
管理 员 登 录 后 台 后 ， 在 功能 导航 浮动 框架 中 选择 “订单 管理 ”/“ 编 辑 订 单 ” 菜 单项 ， 即 可 进入 “ 查 
看 客户 订单 信息 ”页 面 。 该 页 面 的 的 运行 结果 如 图 4.40 所 示 。 


收费 方式 
营 通 干部 
督 通 平 闻 
交通 平邮 
适 货 上 门 
若 通 干部 
交通 平邮 
营 通 于 地 
转 快 专 过 
芹 通 平地 
营 通 平邮 
转 快 专 过 
送 货 上 门 
特快 专 送 
建设 可 行 元 款 | 适 货 上 门 
交通 由 行 汇款 | 转 快 专 遂 


Ee. 


EE 


| 
nl” 


机 
刘 当 
可 


下 到 下 


EE 


地 站 共有 订单 15 条 每 页 


4.40 查看 客户 订单 信息 页 面 的 运行 结果 
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当 用 户 提交 订单 后 ， 系 统管 理 员 就 可 以 通过 “订单 管理 ”模块 下 的 “编辑 订单 ”功能 查看 用 户 已 
经 提交 的 订单 。 在 该 页 面 中 ， 管 理 员 不 仅 可 以 同时 查看 多 个 用 户 的 订单 信息 ， 而 且 可 以 同时 删除 多 个 
订单 。 查 看 客户 订单 信息 的 代码 如 下 : 


例 程 20 ”代码 位 置 : 光盘 \TM\04\ shop\admin\lookdd php 
<table width="750" height="44" border="0" align="center' cellpadding="0" cellspacing="1"> 
<t> 
<td width="121" height="20" bgcolor="#FFFFFF"><div align="center"> 订 单 号 </div></td> 
<!-- 省 略 部 分 关于 订单 标题 的 HTML 标 记 代码 --> 
<td width="115" bgcolor="#FFFFFF"><div align="center"> 操 作 </div></td> 


</t> 

<2php 
include("conn/conn.php"); // 连 接 数据 库 文件 
S$sql=mysql_query("select count(*) as total from tb _dingdan ".$conn): 
S$info=mysql fetch_array($sq]): /检索 订单 数据 表 信息 
Stotal=$info[total]: /计算 用 户 订单 数目 
if($total—0){ // 如 果 订 单数 目 为 0， 则 弹出 相关 提示 

echo "本 站 暂 无 订单 1": 
} 
else{ // 如 果 订 单数 目 不 为 空 ， 则 输出 订单 信息 
dof /应 用 do…while 循 环 语句 输出 订单 信息 
$array=explode("@".Sinfol[spc]): /将 记录 商品 串 存放 到 数组 Sarray 中 
@ Ssum=count(Sarray)*20+260; /设置 弹出 订单 的 高 度 

?> 

<!-- 以 下 代码 用 于 显示 商品 信息 --> 

<t> 


<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[dingdanhao]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[xiadanren]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[shouhuoren]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[total]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[zfff]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo S$infol[shff]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"><?php echo $infol[zt]:?></div></td> 
<td height="21" bgcolor="#FFFFFF"><div align="center"> 
© <input name="button" type="button" class="buttoncss" id="button" 
‘onClick="ijavascript:window.open('showdd.php?id=<?php echo Sinfollid];?>','newframe','width=600,height=<?php 
echo Ssum;?>,left=100,top=100,menubar=no,toolbar=no,location=no,scrollbars=n0")" value=" 查 看 "> 
&nbsp: 
<input name="button2" type="button" class="buttoncss" id="button2" 
onClick="javascript:window.location='orderdd.php?id=<?php echo $info1[id]:?>"…" value=" 执 行 "> 
<input type="checkbox" name=<?php echo $infol[id]:?> value=<?php echo $infol[id]:?>></div></td> 
</tr> 
<2php 
}while($infol=mysql fetch_array($sql1)): //do…while 循 环 语句 结束 
?> 
</table> 


< 代 码 贴 二 
@ $sum=count($array)*20+260:: 用 于 动态 地 改变 弹出 的 订单 的 高 度 ，count($array)*20 表示 所 有 订单 中 商品 项 的 总 
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高 度 ，260 表示 除 所 有 订单 商品 项 以 外 的 高 度 ， 其 中 所 有 两 者 的 和 就 是 弹出 订单 的 总 高 度 ， 这 里 用 $sum 表示 。 以 后 只 
需 将 $sum 的 数值 赋 给 window 的 open() 方 法 的 height 特征 即 可 , 这 样 弹出 订单 的 高 度 将 根据 某 条 订单 商品 数量 的 多 少 而 
增高 或 减低 。 

@ <input...>: 通过 这 段 代 码 中 的 window.open() 方 法 动态 地 弹出 订单 。 

管理 员 具 备 删除 订单 的 操作 权限 ， 由 于 订单 是 电子 商务 平台 的 主线 ， 因 此 ， 订 单一 旦 删除 将 不 能 
恢复 ， 因 此 管理 员 在 操作 时 要 谨慎 。 

选中 和 欲 删除 的 订单 信息 后 面 的 复 选 框 〈 支 持 单条 和 多 条 订单 删除 ) ， 单 击 “ 删 除 选择 项 ”按钮 ， 
即 可 提交 表单 信息 到 数据 处 理 页 deletedd.php， 删 除 指定 的 订单 记录 ， 代 码 如 下 : 


例 程 21 代码 位 置 : 光盘 \TM\04\ shop\admin\lookdd.php 


<?php 
$page=intval($_ POST[page id]): // 获 取 欲 删除 的 订单 号 
include("conn/conn.php"): // 连 接 数 据 库 文件 
while(list($value.$name)=each($ POST)) { // 应 用 while 循 环 语句 ， 删 除 指定 的 订单 信息 


mysql_query("delete from tb_dingdan where id=".$value."",$conn):; 


} 
header("location:lookdd.php?page=".$page.""): // 重 新 定位 到 编辑 订单 页 
> 


4.10.4 执行 客户 订单 信息 的 实现 过 程 


国 查看 客户 订单 信息 使 用 的 数据 表 : tb_dingdan、tb_shangpin 

管理 员 登 录 后 台 后 ， 在 功能 导航 浮动 框架 中 选择 “订单 管理 ”/“ 编 辑 订 单 ” 菜 单项 ， 即 可 进入 “ 查 
看 客户 订单 信息 ”页 面 ， 在 该 页 面 单 击 欲 执行 订单 后 面 的 “执行 ”按钮 ， 进 入 “执行 客户 订单 信息 ” 
页 面 。 该 页 面 的 运行 结果 如 图 4.41 所 示 。 


[cd 已 发 货 也 [Es Md 
注 : 一 纪 商 品 确定 其 贫 。 语 商品 数量 格 目 动作 在 存 中 杠 记 减少 并且 不 可 于 5 


图 4.41 执行 客户 订单 信息 页 面 的 运行 结果 
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执行 客户 订单 是 为 了 改变 订单 的 当前 状态 ， 从 而 使 管理 员 能 够 及 时 有 效 地 处 理 每 个 用 户 的 订单 ， 
并 记录 当前 订单 的 处 理 状态 。 

在 “查看 客户 订单 信息 ”页 面 中 单 击 “ 执 行 ” 按 钮 , 应 用 javascript 脚本 中 的 window 对 象 的 location() 
方法 跳 转 到 orderdd.php 页 ， 并 将 欲 执行 订单 的 id 号 一 并 传递 到 该 页 ， 代 码 如 下 : 


例 程 22 ”代码 位 置 : 光盘 \TM\04\ shop\admin\lookadd.php 


<input name="button2" type="button" class="buttoncss" id="button2" onClick="javascript:window.location='orderdd.php 
?id=<?php echo $info1[id]:?>':" value=" 执 行 "> 

获取 从 查看 客户 订单 信息 页 传递 过 来 的 id 值 , 应 用 mysql_query0 函 数 检索 订单 id 所 对 应 的 详细 信 
息 。 然 后 应 用 explodeO 函 数 将 该 订单 中 商品 的 id 及 这 些 商 品 的 数量 存放 到 数组 中 ， 最 后 通过 for 循环 
语句 显示 该 订单 中 所 有 的 商品 信息 ， 执 行 客户 订单 信息 的 代码 如 下 : 

例 程 23 ”代码 位 置 ; 光盘 \VTMVWO4\ shop\admin\orderdd.php 


<?php 
include("conn/conn.php"): 
S$id=$_GETI[id]: /获取 订单 id 
$sql=mysql query("select * from tb_dingdan where id=".$id."",$conn); 
Sinfo=mysql fetch array($sql); // 从 dingdan 表 中 获取 该 订单 的 详细 信息 
?> 
党 /省 略 了 订单 标题 部 分 的 HTML 标 记 ， 请 参看 本 书 附带 光盘 
<?php 
Sarray=explode("(@",$info[spc]): 
Sarraysl=explode("@",$info[slc]): // 将 该 订单 中 商品 的 id 及 这 些 商 品 的 数量 存放 到 数组 中 
Stotal=0; 
for($i=0:$i<count(Sarray)-1:Si+H){ // 通 过 循环 显示 该 订单 中 所 有 的 商品 信息 


if($array[$i]!=""){ 
$sqll=mysql_query("select * from tb_shangpin where id=".$array[$i]."™,$conn); 
Sinfol=mysql fetch_array($sql]); 
Stotal=$total+ $infol [huiyuanjia]* $arraysl[$i]: 1/ 计算 订单 中 商品 的 总 价格 


?> 
二 涛 最 未 订 单 信息 一 > 
<t> 


<td height="25" bgcolor="#FFFFFF">&nbsp:<?php echo $infol[mingcheng]:?> </td> 

<td height="25" bgcolor="#FFFFFF"><?php if($infol[shuliang]<0) echo " 售 完 ": else echo $arraysl[$i]:?> </td> 

<td height="25" bgcolor="#FFFFFF"><?php echo $infol[shichangjia]:?> </td> 

<td height="25" bgcolor="#FFFFFF"><?php echo $infol[huiyuanjia]:?> </td> 

<td height="25" bgcolor="#FFFFFF"><?php echo $infol[huiyuanjia]:?></td> 

<td height="25" bgcolor="#FFFFFF"><?php echo ceil(($infolfhuiyuanijial/$infol[shichangijial)*100):?>% </td> 

<td height="25" bgcolor="#FFFFFF"><?php echo S$infol[huiyuanjia]*$arraysl[$i]:?> </td> 
</t> 
< --------------------- 一 a 一- 一- -一 一 ~ ee — a > 
<?php 

} 
/ 


?> 


<!-- 收 货 人 信息 显示 部 分 代码 略 --> 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


选中 “已 收 款 ”、“ 已 发 货 ”、“ 已 收 货 ” 复 选 框 ， 单 击 “ 修 改 ”按钮 ， 提 交 订 单 的 处 理 状态 到 
数据 处 理 页 saveorderphp， 更 新 订单 表 的 处 理 状态 ， 并 对 商品 信息 表 中 的 商品 数量 进行 相应 减少 ， 限 
于 篇 幅 ， 详 细 代码 请 参看 本 书 附带 光盘 。 


4.10.5 打印 客户 订单 信息 的 实现 过 程 


国 查看 客户 订单 信息 使 用 的 数据 表 : tb_dingdan、tb_shangpin 

管理 员 登 录 后 台 后 ， 在 功能 导航 浮动 框架 中 选择 “订单 管理 ”/“ 编 辑 订 单 ” 菜 单项 ， 即 可 进入 “ 查 
看 客户 订单 信息 ”页 面 ， 在 该 页 面 单 击 欲 执行 订单 后 面 的 “查看 ”按钮 ， 进 入 “打印 客户 订单 信息 ” 
页 面 ， 客 户 订单 及 打印 预览 页 面 的 运行 结果 如 图 4.42 和 图 4.43 所 示 。 


加 http://127.0.0.1 - 商品 订单 - Microsoft Intei [elF 
单 号 : 2007112214363739 打印 预览 ”| 打印 
崎 品 列表 下 ) : 
数码 相机 2699 2599 1 2599 
总 计 和 费用 :2599| 

下 单 人 : 纯净 水 

收 货 人 : 紫 政 

收 货 人 地 址 : ”吉林 省 长 春 市 河东 路 1** 号 

邮 编 : 130000 电 话 : 。 0431-649T8rty 

-mail ixuanyyBsi com 

送 货 方式 : 。 送 货 上 门 支付 方式 : 网 上 支付 

请 您 在 一 周 内 按 烙 的 支付 方式 进行 汇 协 , 汇款 时 注 明 烽 的 订单 号 ! 汇 款 后 请 及 时 通知 我 们 

XH | a ; 2007-11-22 14:36:31 


图 4.42 客户 订单 页 面 的 运行 结果 


打印 @. | 攻 | * “而 WW 人 共 ! 一 汪 | 节 分 Jiomw 可 | 各 中 
商品 订单 
订单 号 : 2007112214363739 
商品 列表 下 ) : 
商品 名 称 市 场 价 会 员 价 数量 小 计 
数码 相机 2699 2599 1 2599 
总 计 筑 用 :2599 
下 单 人 : 纯净 水 
收 货 人 : 茶 洲 
收 货 人 地 址 : 衣 林 省 长 春 市 河东 路 1#* 号 
邮 蝙 : 130000 电 话 : 0431-8497Bee+ 
Enail ixaantyBsi com 
送 货 方式 : 。 送 货 上 门 支付 方式 ; 网 上 支付 
请 您 在 一 周 内 按 您 的 支付 方式 进行 汇款 , 汇款 时 注 明 您 的 订单 号 ! 汇 款 后 请 及 时 通知 我 们 


他 陵 时 间 : 2007-11-22 14:36:37 证 
FE OO 瑞 


图 4.43 客户 订单 打印 预览 页 面 的 运行 结果 
在 电子 商务 平台 中 加 入 打印 客户 订单 的 功能 ， 不 但 可 以 使 用 户 非常 方便 地 操作 程序 ， 提 高 工作 效 
率 ， 而 且 更 能 使 程序 适应 人 性 化 的 潮流 。 
在 “查看 客户 订单 信息 ”页 面 ， 单 击 “ 查 看 ”按钮 ， 应 用 JavaScript 脚本 中 的 window 对 象 的 open0 
方法 打开 showdd.php 页 ， 并 将 欲 打印 订单 的 id 号 一 并 传递 到 该 页 ， 打 印 指定 id 的 订单 信息 ， 代 码 


和 
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如 下 : 


例 程 24 ”代码 位 置 : 光盘 \TMW4\ shop\admin\lookadd.php 

<input name="button" type="button" class="buttoncss" id="button" onClick="javascript:-window.open('showdd.php?id= 
<?php echo Sinfol[id];?>','newframe',"width=600,height—<?php echo $sum:?>,left=100.,top=100,menubar=no,toolbar=no,location 
=no,scrollbars=no")" value=" 查 看 "> 


打印 客户 订单 信息 页 面 主要 应 用 WebBrowser+CSS 方法 实现 客户 订单 的 打印 ， 该 方法 简单 、 方 便 、 


快捷 ， 在 浏览 网 页 的 同时 就 可 以 实现 打印 及 打印 预览 功能 。 实 现 客户 订单 信息 打印 及 打印 预览 的 方法 
如 下 : 


例 程 25 ”代码 位 置 ; 光盘 \TMVWO4\ shop\admin\ showdd.php 
<! 一 定义 CSS 样 式 --> 
<style type="text/css"> 
@ Mmediaprint{ 
div{display:none} 
} 


</style> 

<div align="right"> <!-- 添 加 div 层 --> 

<script> 

function prnO{ /定义 prmn0 函 数 调 用 ExecWB 方 法 实现 打印 预览 功能 


document.all. WebBrowserl.ExecWB(7.,1) 
} 


</script> 
© <objectID="WebBrowserl’ WIDTH=0 HEIGHT=0 
CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></obiect> 

<input type="button" value=" 打 印 预 览 " class="buttoncss" onclick="prnO">&nbsp: 
®@ <inputtype="button" value=" 打 印 " class="buttoncss" onClick="window.printO"> 

</div> 


< 八代 码 贴 寺 

@ @media print{div{display:none}}: 定义 打印 样式 为 不 显示 状态 。 该 功能 在 设计 表单 时 将 按钮 及 表格 均 添加 在 div 
层 中 ， 然 后 将 所 有 应 用 div 层 的 内 容 应 用 该 样式 ， 这 样 在 打印 及 打印 预览 时 div 层 的 内 容 不 打印 ， 如 图 4.35 所 示 。 

@ <object…></object>: 是 插入 的 ActiveX 组 件 ， 目 的 是 将 系统 的 打印 预览 功能 嵌入 该 页 。 函数 pmO 中 的 document 
.all.WebBrowserl.ExecWB(7.1) 是 调用 该 组 件 的 ExecWB 方法 ， 从 而 实现 打印 预览 功能 。 

@ window.print(0): 在 单 击 事件 下 调用 window 对 象 的 print0 方 法 实现 的 打印 功能 。 
[加 说 明 : 关闭 窗口 是 通过 调用 window 对 和 象 的 close() 方 法 来 实现 的 。 如 果 用 户 浏览 器 禁用 了 Active 

组 件 ， 则 打印 预览 功能 可 能 无 法 实现 。 


4.10.6 ”查找 客户 订单 信息 的 实现 过 程 
国 ”查找 客户 订单 信息 使 用 的 数据 表 : ”tb_dingdan 


管理 员 登 录 后 人 台 后 ,在 功能 导航 浮动 框架 中 选择 “订单 管理 ”/“ 查 询 订单 ”菜单 项 ， 即 可 进入 “ 查 
找 客户 订单 信息 ”页 面 ， 该 页 面 的 的 运行 结果 如 图 4.44 所 示 。 
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订 音 叶 2007112214385739 
EE 
订单 号 下 晶 用 户 可 计 人 | 全 额 对 [ 付款 方式 | 收 吉方 式 | 订单 在 
22 Er 58。 | 同上 支付 | 这 党 上门 | 末 作 任 问 外 可 
Er 


图 4.44 查找 客户 订单 信息 页 面 的 运行 结果 


为 了 便于 系统 管理 员 管理 订单 ， 该 系统 后 台 提 供 了 管理 员 订 单 查找 功能 ， 管 理 员 可 按 下 订单 人 姓 
名 或 订单 号 查询 订单 的 相关 信息 。 
查找 订单 信息 页 面 涉及 到 的 HTML 表单 的 重要 元 素 如 表 4.2 所 示 。 


表 4.2 查找 订单 信息 页 面 涉 及 到 的 HTML 表单 的 重要 元 素 


名 称 含义 
Re 查询 订单 表单 
onSubmit="return chkinput3( this)” 
semame F 订 单 人 姓名 
adh 订单 号 
show find 隐藏 域 
buton | smil | cas-bocw | “查找 ”按钮 


输入 查询 条 件 ， 单 击 “ 查 找 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 理 页 ， 对 指定 条 件 的 订单 信息 进行 检 
索 。 通 过 让 条 件 语句 进行 判断 ， 如 果 下 订单 人 姓名 为 空 ， 则 按 订 单 号 查询 ;如果 订单 号 为 空 ， 则 按 下 
订单 人 姓名 查询 :如果 两 者 均 不 为 定 ， 则 按 以 上 两 个 条 件 同时 查询 。 最 后 ,应 用 do…while 循环 语句 输 
出 与 查询 条 件 所 匹配 的 订单 信息 。 关 键 代码 如 下 : 


例 程 26 ”代码 位 置 : 光盘 \TMVWO4\ shop\admin\finddd.php 


<2?php 
if($_POST[show_find]!=""){ // 如 果 表 单 被 提交 则 开始 查找 订单 信息 
$username=trim($_ POST[username]): /获取 下 订单 人 姓名 
Sddh=trim($_ POST[ddh]): // 获 取 订 单 号 
if(Susername—""){ // 如 果 下 订单 人 姓名 为 空 ， 则 按 订 单 号 查询 


$sql=mysql_query("select * from tb_dingdan where dingdanhao=".$ddh."",$conn): 


} 
elseif(Sddh—""){ /如 果 订 单 号 为 室 ， 则 按 下 订单 人 姓名 查询 
S$sql=mysql_query("select * from tb _ dingdan where xiadanren=". $username."™".$conn); 
} 
else{ /如 果 两 者 均 不 为 空 ， 则 按 以 上 两 个 条 件 同时 查询 
$sql=mysql_query("select * from tb_dingdan where xiadanren=".Susermame."and dingdanhao=".$ddh."".$conn): 


> 
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} 
Sinfo=mysql] fetch array($sq]l): 
这 $info 一 false){ /如 果 无 记录 ， 则 弹出 相关 的 提示 信息 
echo "<div algin='center> 对 不 起 ,没有 查找 到 该 订单 !</div>": 
} 


else{ // 否 则 ， 按 指定 的 查询 条 件 输出 订单 信息 
一 /订单 标题 部 分 HTML 标记 代码 略 ， 请 参见 光盘 
/应 用 do…while 循 环 语句 输出 指定 条 件 下 的 订单 信息 
全 


<td bgcolor="#FFFFFF"><div align="center"><?php echo $info[dingdanhao]:?></div></td> 
/显示 订单 详细 内 容 部 分 代码 略 ， 请 参见 光盘 
</t> 
<?php 
}while(Sinfo=mysql_fetch_array(SsqD)); 
} 
} 


?> 


4.11 开发 技巧 与 难点 分 析 


4.11.1 防止 非法 用 户 绕 过 系统 登录 直接 进入 系统 


为 了 保证 网 站 内 信息 资源 的 安全 , 项 目 开 发 人 员 应 防止 用 户 以 非法 身 


份 对 网 站 内 部 信息 进行 非法 操作 , 只 有 通过 正确 的 途径 成 功 登录 电子 商务 AR 


而 直接 单 击 “ 我 的 购物 车 ”或 “购买 ” 某 项 商品 ， 则 强制 跳 到 首页 进行 合 
录 操 作 ， 运 行 结果 如 图 4.45 所 示 。 


台 , 才 可 以 进行 商品 的 添加 及 操作 购物 车 列表 等 操作 。 如 果 用 户 未 登录 


下 面 讲解 开发 该 功能 的 编程 思路 。 
首先 ， 在 用 户 登录 时 应 用 session 变量 记录 用 户 名 ， 即 $_ SESSION[username]。 
然后 ， 对 登录 的 $ SESSION[userame] 进 行 判断 ， 如 果 $_SESSION[username] 值 为 空 将 弹出 提示 框 ， 


如 图 4.45 所 示 ， 并 强制 将 页 面 跳 转 到 电子 商务 平台 原来 页 面 进行 用 户 登 录 。 其 代码 如 下 : 


原来 页 面 


<?php 
session start(): // 初 始 化 session 变 量 
if($_SESSION[usemame]—"") { // 判 断 用 户 是 否 已 经 登录 
echo "<script>alert(' 请 先 登录 后 购物 !"):history.back0:</script>": /如 果 用 户 未 登录 ， 则 提示 用 户 先 登录 并 返回 到 
exit: // 用 exit 语 句 停止 循环 的 继续 执行 
> 
?> 
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准 技巧 : 为 了 提高 代码 的 可 重用 性 ， 开 发 人 员 可 以 将 上 面 这 段 代码 独立 封装 在 PHP 文件 中 ， 并 将 广 
件 命名 ， 如 check_user.php， 然 后 将 该 文件 应 用 include 包含 语句 嵌入 在 必须 通过 登录 才能 
访问 的 页 面 中 。 如 果 用 户 未 进行 登录 ， 而 直接 通过 地 址 栏 或 其 他 手段 访问 该 页 面 时 ， 则 将 
会 弹出 用 户 先 登录 方 可 有 访问 权限 ， 并 强制 跳 转 到 登录 页 。 应 用 include 包含 语句 府 入 该 文 
件 的 代码 如 下 : 
<?php include "check user.php":?> 


[DD 说 明 : 利用 引用 语句 来 包含 其 他 文件 ， 以 减少 代码 的 重复 ， 是 PHP 编程 的 重要 技巧 。 关 于 引用 语 
多 的 详细 讲解 ， 请 读者 参见 4.7.2 节 相 关内 容 。 


4.11.2 ”检测 用 户 名 是 否 已 经 注册 


在 电子 商务 平台 中 ， 为 了 提高 网 站 的 安全 性 ， 在 实现 会 员 注册 功能 模块 时 ， 要 求 不 允许 同时 存在 
两 个 或 多 个 相同 的 用 户 名 。 

在 “用 户 昵称 ”文本 框 中 输入 用 户 名 ， 通 过 单 击 “ 检 测 用 户 ” 超 链接 对 用 户 输 入 的 新 昵称 进行 检 
测 ， 并 输出 检测 结果 ， 运 行 结果 如 图 4.46 所 示 。 


4.46 检测 用 户 名 是 否 已 被 注册 


下 面 讲解 开发 该 功能 的 编程 思路 : 
(1) 在 会 员 注册 页 面 添加 “查看 昵称 是 否 已 用 ”按钮 ， 其 HTML 标记 的 代码 如 下 : 
<input name="button2” type="button” onclick="chknc(forml.usernc.value)” value=" 查 看 昵称 是 否 已 用 "> 


(2) 在 该 按钮 的 单 击 事件 下 调用 自 定 义 的 chknc0O 函 数 ， 通 过 window.open 方法 打开 一 个 用 户 检 
测 的 网 页 对 话 框 ， 代 码 如 下 : 


‘<script language="javascript"> 
function chknc(nc) { 
window.open("chkusemc.php?nc="+nc."newframe"."width=200.height=10.left=500.top=200.menubar=no.toolbar=no.loca 
tion=no.scrollbars=no.location=n0"): 
} 
</script> 
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(3) 将 输入 的 用 户 昵 称 通过 Get0 方 法 提交 到 chkusernc.php 数据 处 理 页 ， 判 断 新 注册 的 用 户 昵 称 
是 否 为 空 ， 如 果 为 空 ， 弹 出 提示 信息 ; 否则， 检测 该 昵称 是 否 已 被 占用 。 如 果 占 用 ， 则 弹出 提示 信息 ; 
否则 提示 用 户 ， 该 昵称 可 以 注册 ， 代 码 如 下 : 


<?php 
include("conn/conn.php"); // 连 接 数 据 库 文件 
Snc=trim($ GET[nc]); // 获 取 用 户 昵称 
if($nc—""){ // 如 果 用 户 昵称 为 空 ， 则 弹出 提示 信息 
echo "请 输入 昵称 "; 
} 
else{ /否则 ， 检 测 用 户 昵称 是 否 被 占用 
Ssql=mysql query("select * from tb user Where name="".Snc.""",Sconn); 
Sinfo=mysql_fetch_array(SsqD; /检索 数据 信息 
这 $info 一 tmue){ /如果 用 户 表 中 该 昵称 已 存在 ， 则 弹出 提示 信息 
echo "对 不 起 ,该 昵称 已 被 占用 !…: 
} 
else{ // 如 果 该 昵称 不 存在 ， 说 明 可 以 注册 
echo "恭喜 ,该 昵称 没 被 占用 !”: // 弹 出 提示 信息 


} 
} 


?> 


4.11.3 ”用 户 安全 退出 


为 了 保证 网 站 的 安全 ， 在 用 户 完成 购物 后 ， 需 要 安全 退出 电子 商务 平台 ， 需 单 击 “ 注 销 离开 ” 超 
链接 ， 即 可 在 退出 网 站 时 清空 session 变量 ， 代 码 如 下 : 


<?php 


session _startO; /启动 会 话 
Session_unsetO; /删除 会 话 
Session_destroy0; // 结 束 会 话 
header("location: index.php"): // 重 新 定位 到 首页 


?> 


4.12 加密 技术 专题 


加 密 是 在 不 安全 的 信息 渠道 中 实现 信息 安全 传输 的 一 种 重要 方法 。 因 此 对 数据 进行 加 密 操作 是 非 
常 重要 的 。PHP 中 加 密 的 方法 有 很 多 种 ， 开 发 人 员 在 应 用 加 密 技术 时 ， 可 根据 情况 进行 选择 性 使 用 。 
下 面 对 各 项 加 密 技术 进行 详细 讲解 。 
4.12.1 URL 编码 加 密 技 术 

PHP 通过 GET( 方 法 提交 的 信息 ， 会 以 查询 字符 串 的 形式 显示 在 浏览 器 的 地 址 栏 中 ， 这 样 会 给 网 
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站 的 安全 带 来 很 大 的 隐患 ， 为 了 解决 这 个 问题 ， 可 以 对 查询 字符 串 进行 URL 编码 。 

PHP 中 实现 对 查询 字符 串 进 行 URL 编码 可 以 通过 urlencode0 函 数 实现 。 该 函数 的 使 用 格式 如 下 : 

string urlencode( string str) 

该 函数 实现 将 字符 串 str 进行 URL 编码 。 

PHP 中 实现 对 URL 编码 后 的 查询 字符 串 进行 解码 ， 通 过 urldecode0 函 数 实现 。 该 函数 的 使 用 格式 
如 下 : 

string urldecode( string str) 

该 函数 实现 将 URL 编码 str 进行 解码 。 

下 面 应 用 urlencode0 函 数 对 “首页 ”字符 串 进行 URL 编码 ， 代 码 如 下 : 

<a href=index.php?page<?php echo urlencode(" 首 页 ")?> > 首页 </a> 

接收 查询 字符 串 的 值 ， 并 应 用 urldecode0 函 数 对 查询 字符 串 进行 解码 ， 代 码 如 下 : 

<2php 


echo urldecode($_GET[page]): // 对 GET 方 法 提交 的 字符 串 进 行 URL 解 码 
人 > 


4.12.2 ”base64 编码 加 密 技术 


PHP 实现 字符 串 的 base64 编码 通过 base64_encode0 函 数 实现 。 该 函数 的 语法 格式 如 下 : 

string base64 encode(string data) 

参数 data 指 要 进行 base64 编码 的 数据 。 该 函数 的 返回 结果 为 字符 串 类 型 。 

PHP 实现 对 base64 编码 的 字符 进行 解码 , 通过 base64_decode0) 函 数 实现 。 该 函数 的 语法 格式 如 下 : 
string base64_decode (string encoded data) 

参数 encoded data 是 指 要 进行 base64 解码 的 字符 串 。 

下 面 应 用 base64 编码 讲解 加 密 和 解密 的 过 程 ， 代 码 如 下 : 


<?php 
Spwd=111: // 定 义 加 密 的 密码 
echo base64_encode(Spwd): // 返 回 加 密 后 的 值 “MTEx” 
Sjiami=base64_encode($pwd): // 将 加 密 后 的 字符 赋 给 变量 
echo base64 decode(S$jiami): // 对 加 密 的 字符 串 进行 解密 ， 返 回 值 为 “111” 
> 


4.12.3 ”crypt() 加 密 技术 


在 Web 程序 开发 过 程 中 ， 可 以 应 用 PHP 提供 的 crypt0 函 数 来 完成 加 密 功 能 。crypt0 函 数 是 单 向 的 加 
密 函数 , 无 法 解密 。 经 过 加 密 的 口令 即使 被 非法 获取 , 由 于 不 能 被 还 原 为 明文 , 也 不 会 影响 网 站 的 安全 性 。 
crypt0 函 数 的 语法 格式 如 下 : 
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string crypt(string str, string [salt]); 

参数 说 明 : 

str: 是 需要 加 密 的 明文 字符 串 。 

回 salt; 是 一 个 位 字 串 ， 能 够 影响 加 密 的 暗 码 ， 进 一 步 排除 被 破解 的 可 能 性 。 默 认 情 况 下 ， 位 字 
串 的 长 度 为 2 位 。 若 不 使 用 salt 参数 ， 则 程序 会 自动 产生 干扰 串 。 

下 面 应 用 crypt0 加 密 函 数 对 字符 串 进 行 加 密 ， 代 码 如 下 : 


<2?php 
Spassword="111"; // 定 义 加 密 的 密码 
$salt=substr($password,0.2): // 计 算 位 字 串 的 数值 
echo Spassword=crypt($password,$salt): // 返 回 加 密 后 的 值 “11I5m18GD4M5g” 
?> 


4.12.4 MD5 加 密 技术 


MD5 是 在 Web 应 用 程序 中 最 常用 的 密码 加 密 算 法 之 一 ， 它 和 crypt0 函 数 一 样 也 是 单项 加 密 的 。 

MD5 广泛 用 于 数据 加 密 技术 上 ， 在 很 多 网 站 中 ， 用 户 的 密码 是 以 MD5 值 的 方式 保存 的 ， 用 户 登 
录 时 ， 程 序 员 把 用 户 输入 的 密码 计算 成 MD5 值 ， 然 后 再 去 和 数据 库 中 保存 的 MD5 值 进行 比较 ， 而 程 
序 本 身 并 不 “知道 ”用 户 的 密码 的 真实 值 ， 从 而 提高 了 网 站 的 安全 性 。 

md50 函 数 用 来 计算 字符 串 的 MD5 混合 值 。 其 语法 如 下 : 

string mds(string str); 


电子 商务 平台 网 后 人 台 管 理 员 登录 模块 即 采用 md50 函 数 获 取 字 符 串 加 密 后 的 值 。 加 密 管理 员 密 码 的 
部 分 代码 如 下 : 


<?php 

Sstr="111"; // 管 理 员 密码 “111” 

Smstr=md5($str): // 将 管理 员 密 码 通过 md50 函 数 加 密 

echo $mstr // 返 回 加 密 后 的 值 “98d51a19d8al21ce581499d7b701668” 
?> 


全 注意 :; 电子 商务 平台 网 后 台 管 理 员 登 录 页 面 采用 了 MDS 加 密 技 术 ， 具 体 代码 请 读者 参见 本 书 附 
赠 光 盘 TM\04\shop\admin\index.php 文件 。 


4.13 本 章 总 结 


本 章 运用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 电子 商务 平台 网 带领 读者 详细 讲解 一 个 系统 的 开 
发 流程 。 同 时 ， 在 电子 商务 平台 网 的 开发 过 程 中 ， 采 用 了 浮动 框架 技术 ， 使 整个 系统 的 设计 思路 更 加 
清晰 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 以 了 解 一 般 网 站 的 开发 流程 ， 而 且 可 以 熟悉 购物 车 、 订 单 及 加 密 
技术 的 开发 思想 。 


第 章 


网 络 在 线 考 试 系统 


(Apache+PHP+Ajax+SQL Server 2000 实现 ) 
( 婴 视频 讲解 : 1 小 时 23 分钟) 


随 着 科技 的 发 展 ， 网 络 技术 已 经 深入 到 人 们 的 日 常生 活 中 ， 同 时 带 来 了 教育 方 
式 的 一 次 变革 。 而 网 络 考试 则 是 一 个 很 重要 的 方向 。 基 于 Web 技术 的 网 络 考试 系 
统 可 以 借助 于 遍布 全 球 的 Internet 进行 ， 因 此 考试 既 可 以 在 本 地 进行 ， 也 可 以 在 异 
地 进行 ， 从 而 大 大 拓展 了 考试 的 灵活 性 。 并 且 缩 短 了 传统 考试 要 求 老 师 打 印 试 卷 、 
安排 考试 、 监 考 、 收 集 试 卷 、 评 改 试 卷 、 讲 评 试卷 和 分 析 试 卷 ， 这 个 漫长 而 复杂 的 
过 程 ， 使 考试 更 趋 于 客观 、 公 正 。 本 章 介 绍 了 一 个 具有 在 线 考 试 、 即 时 阅卷 、 成 绩 
查询 以 及 考题 和 考生 信息 管理 等 功能 的 网 络 在 线 考试 系统 。 

通过 阅读 本 章 ， 读 者 可 以 : 


掌握 网 络 在 线 考试 的 开发 过 程 

掌握 应 用 SQL Server 2000 创建 数据 库 、 数 据 表 
掌握 连接 SQL Server 数据 库 的 不 同方 式 

实时 显示 时 间 的 两 种 不 同 的 方法 

掌握 Ajax 无 刷新 技术 


各 吾 吾 吾 至 
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S$.1 开发 背景 


随 着 教育 改革 的 不 断 深入 ， 近 几 年 来 出 现 了 一 些 新 的 教育 形式 ， 如 网 上 授课 、 网 上 考试 等 。X X 
高 校 是 一 所 师资 力量 非常 雄厚 的 学 校 ， 随 着 院 校 的 扩招 ， 学 生 数 量 不 断 增加 。 为 了 适应 新 形式 的 发 展 ， 
改变 传统 的 教学 模式 ， 方 便 学 生 随时 随地 的 对 自己 的 学 习 情况 进行 检测 ， 减 轻 教 师 的 工作 压力 ， 现 委 
托 其 他 公司 开发 网 上 在 线 考试 系统 。 这 种 无 纸 化 的 网 络 考试 系统 ， 使 考 务 管理 突破 时 空 限制 ， 提 高 了 
考试 工作 效率 和 标准 化 水 平 ， 使 学 校 管理 者 、 教 师 和 学 生 在 任何 时 候 、 任 何 地 点 均 可 以 通过 网 络 进行 
考试 。 网 络 在 线 考试 系统 已 经 成 为 教育 技术 发 展 与 研究 的 方向 。 


5.2 系统 分 析 


5.2.1 需求 分 析 


随 着 计算 机 技术 的 发 展 和 网 络 技术 的 日 益 成 熟 ， 通 过 网 络 进行 信息 交流 已 成 为 一 种 快捷 的 交互 方 
式 。 在 这 种 网 络 环境 下 ， 学 校 或 考试 机 构 希 望 通 过 建立 网 络 在 线 考试 网 站 来 扩大 知名 度 、 降 低 管理 成 
本 和 减少 人 力 物力 的 投资 ， 从 而 为 考生 提供 更 全 面 、 更 灵活 的 服务 ， 并 全 面 、 准 确 地 对 考试 进行 跟踪 
和 评价 。 与 此 同时 ， 考 生 希 望 根据 自己 的 学 习 情况 进行 测试 ， 并 能 够 得 到 客观 、 科 学 的 评价 ; 教务 人 
员 希 望 能 够 有 效 地 改进 现 有 的 考试 模式 ， 提 高 考试 效率 。 

通过 实际 情况 的 调查 ， 要 求 网 络 在 线 考试 系统 具有 以 下 功能 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵活 ， 树 立 企业 形象 。 

要 求实 现在 线 考试 功能 ， 自 动 核算 考试 成 绩 。 

要 求 提供 考试 时 间 倒 计时 功能 ， 使 考生 实时 了 解 考试 剩余 时 间 。 
要 求 系统 自动 阅卷 ， 保 证 考试 成 绩 真实 有 效 。 

要 求 考生 凭 准 考证 号 查询 考试 成 绩 ， 以 保证 信息 安全 。 

系统 运行 稳定 、 安 全 可 靠 。 

要 求 对 考生 及 考题 信息 进行 严格 管理 。 


办 办 欠 欠 多 多 轨 


5.2.2 可 行 性 分 析 


可 行 性 分 析 的 目的 就 是 要 用 最 小 的 代价 在 尽 可 能 短 的 时 间 内 确定 问题 是 否 能 够 解决 。 通 过 分 析 解 
法 的 利 浆 ,来 判定 系统 目标 和 规模 是 否 现实 ， 系 统 完成 后 所 能 带 来 的 效益 是 否 达 到 值得 去 投资 开发 这 
个 系统 的 程度 。 网 络 在 线 考 试 系统 的 可 行 性 可 从 以 下 两 方面 考虑 。 

1. 经 济 可 行 性 

定期 地 组 织 考试 是 各 个 院 校 及 时 掌握 学 生 学 习 成 绩 的 有 效 方式 ， 利 用 网 络 在 线 考试 系统 ， 一 方面 
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可 以 节省 人 力 资源 ， 降 低 考试 成 本 。 另 一 方面 ， 在 线 考试 系统 能 够 快速 进行 考试 和 评分 ， 体 现 出 考试 
的 客观 与 公正 性 。 

2. 技术 可 行 性 

开发 一 个 网 络 在 线 考试 系统 ， 涉 及 到 的 最 核心 的 技术 问题 就 是 如 何 实现 在 不 刷新 页 面 的 情况 下 实 
时 显示 考试 时 间 及 剩余 时 间 ， 并 做 到 到 达 考 试 结束 时 间 时 自动 提交 试卷 的 功能 。 通 过 Ajax 技术 可 以 轻 
松 实现 这 些 功能 ， 这 为 网 络 在 线 考试 系统 的 开发 提供 了 技术 保障 。 


5.3 系统 设计 


5.3.1 系统 目标 


根据 前 面 所 作 的 需求 分 析 及 用 户 的 需求 可 知 ， 网 络 在 线 考试 系统 属于 中 小 型 软件 ， 在 系统 实施 后 ， 
应 达到 以 下 目标 : 
采用 开放 、 动 态 的 系统 架构 ， 加 强 用 户 与 网 站 的 动态 交互 性 。 
具有 空间 性 。 被 授权 的 用 户 可 以 在 异地 登录 网 络 在 线 考试 系统 ， 无 须 到 指定 地 点 进行 考试 。 
操作 简单 方便 、 界 面 简洁 美观 。 
系统 提供 考试 时 间 倒计时 功能 ， 使 考生 实时 了 解 考 试 剩余 时 间 。 
随机 抽取 试题 。 
实现 自动 提交 试卷 的 功能 。 当 考试 时 间 到 达 规 定时 间 时 ， 如 果 考 生还 未 提交 试卷 ， 系 统 将 自 
动 交卷 ， 以 保证 考试 严肃 、 公 正 地 进行 。 
系统 自动 阅卷 ， 保 证 成 绩 真 实 准确 。 
考生 可 以 查询 考试 成 绩 。 
对 考生 注册 信息 进行 管理 。 
系统 运行 稳定 、 安 全 可 靠 。 


5.3.2 系统 功能 结构 


办 办 办 多 办 凶 


办 办 办 办 


网 络 在 线 考试 系统 的 前 台 功 能 结构 如 图 5.1 所 示 。 
网 络 在 线 考试 系统 的 后 台 功 能 结构 如 图 5.2 所 示 。 


5.3.3 ”系统 流程 图 
为 了 读者 能 够 更 好 地 学 习 ， 下 面 给 出 网 络 在 线 考试 网 的 网 站 流程 概述 及 系统 流程 图 。 
考生 在 网 络 在 线 考试 系统 中 通过 注册 为 网 站 用 户 ， 登 录 网 站 进行 相关 操作 。 考 生 登 录 后 ， 可 以 进 


行 在 线 考试 、 查 询 成 绩 和 修改 个 人 密码 的 操作 。 在 考试 前 ， 考 生 需 要 阅读 考试 规则 、 选 择 考 试 套 题 后 
开始 考试 ; 考试 时 间 结束 时 ， 考 生 提交 试卷 提交 试卷 的 同时 ， 系 统 将 自动 返回 本 次 考试 的 考试 结果 。 
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网 站 管理 员 通 过 登录 模块 可 以 登录 到 网 站 的 后 台 系统 ， 对 考生 信息 、 考 试 信息 、 管 理 员 信息 进行 管理 
网 络 在 线 考试 系统 的 系统 流程 如 图 5.3 所 示 。 


网 络 在 线 考试 系统 的 前 台 
| 


Lk 


网 络 在 线 考试 系统 的 后 台 


人 

二 考 | | 考 | | 考 | | 考 | | 退 

考 县 贯 | | 生 | | 是 | | 题 | | 是 | | 出 
试 考 入 言 | | 类 言 | | 信 | | 后 
规 斌 点 | | 县 | | 更 | | 息 | | 及 | | 各 
则 成 答 | | 管 | | 管 | | 添 | | 管 | | 管 
绩 是 | | 理 | | 理 | | 加 | | 理 | | 理 

图 5.1 网 络 在 线 考试 系统 的 前 台 功 能 结构 图 图 5.2 网络 在 线 考试 系统 的 后 台 功能 结构 图 


自动 阅卷 并 显 
示 考 试 成 绩 


图 5.3 网 络 在 线 考试 系统 流程 图 
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5.3.4 系统 预览 


网 络 在 线 考试 系统 由 多 个 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 可 参见 光盘 中 的 源 程序 。 

考生 登录 页 面 如 图 5.4 所 示 , 该 页 面 主要 用 于 考生 登录 , 实现 在 线 考试 及 考试 成 绩 查 询 等 功能 。 在 
线 考试 页 面 如 图 5.5 所 示 , 该 页 面 用 于 实现 在 线 答题 功能 , 同时 提供 了 显示 考试 时 间 和 剩余 时 间 及 自动 
提交 试卷 功能 。 


人 ac 和 


Nr 


| 

| I 
| 
图 5.4 考生 登录 页 面 ( 光 扒 \TM\05\online ks\dengluphp ”图 5.5 在 线 考试 页 面 (光盘 VTMWS\Vonline_ks\ksks.php) 


考题 类 别管 理 页 面 如 图 5.6 所 示 , 该 页 面 主要 用 于 实现 显示 考题 类 别 的 基本 信息 、 添 加 考题 类 别 和 
删除 考题 类 别 等 功能 。 考 题 信息 管理 页 面 如 图 5.7 所 示 ， 该 页 面 主要 用 于 管理 考试 题目 信息 。 


图 5.6 考题 类 别管 理 (光盘 \TM\05\…\admin\ktlb gl.php) 5.7 考题 信息 管理 (光盘 \TM\05\…\admin\ktxx_gl.php) 
[说明 : 由 于 路 径 太 长 ， 因 此 省 略 了 部 分 路 径 ， 省 略 的 路 径 是 : online ks\。 


”> 
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5.3.5 ”开发 环境 


在 开发 网 络 在 线 考试 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
。 服务 器 端 
操作 系统 : Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.2。 
PHP 软件 : PHP 5.1.6。 
数据 库 : SQL Server 2000。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 ， 最 佳 效果 1024X768 像素 。 
客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 1024X768 像素 。 


园 罗网 网 加 网 加 一 


回回 


5.3.6 ”文件 夹 组 织 结构 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 〈 例 如， 创建 一 个 名 为 images 的 文 
件 夹 ， 用 于 保存 网 站 中 所 使 用 的 图 片 )》 ， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 
架构 。 笔 者 在 网 络 在 线 考 试 系统 中 设计 了 如 图 5.8 所 示 的 文件 夹 架构 图 。 在 开发 时 ， 只 需要 将 所 创建 的 
文件 保存 在 相应 的 文件 夹 中 即 可 。 


日 筷 online_ks 
四 和 in 一 一 一 一 一 一 一 一 一 一 用 于 存 钱 网 站 后 文件 
机 因 om 一 一 一 一 一 一 一 一 一 一 用 于 存 伟 数 据 库 这 接 文件 
日 全 imess 一 一 一 一 一 一 一 一 一 一 用 于 存储 网 站 图 片 资源 
日 个 证 一 一 一 一 一 一 一 一 一 一 用 于 存 依 网 站 使 用 的 自 定义 函数 


图 5.8 ”网 络 在 线 考试 系统 的 文件 夹 组 织 结构 
5.4 数据 库 设计 


5.4.1 数据 库 分 析 


由 于 网 络 在 线 考试 系统 对 于 数据 的 安全 性 及 完整 性 要 求 比较 高 ， 并 且 为 了 增加 程序 的 适用 范围 ， 
还 要 保证 系统 可 以 拥有 存储 足够 多 数据 的 能 力 。SQL Server 2000 是 一 种 高 性 能 的 关系 型 数据 库 管理 系 
统 ， 不 仅 安全 可 靠 而 且 易 于 操作 ， 已 成 为 在 线 事 务 进程 和 数据 仓库 等 最 好 的 数据 库 平台 之 一 。 综 上 所 


Fy 
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述 ， 本 系统 采用 SQL Server 2000 数据 库 。 
5.4.2 ”数据 库 概 念 设计 


根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 使 用 的 数据 库 实体 分 别 为 考生 信 
息 实 体 、 管 理 员 实体 、 考 题 类别 实 体 和 考题 信息 实体 。 下 面 将 介绍 这 几 个 实体 的 E-R 图 。 

1. 考生 信息 实体 

考生 信息 实体 包括 编号 、 考 生 姓名 、 联 系 方式 、 准 考证 号 、 考 试 成 绩 、 考 题 类 别 、 考 试 时 间 、 联 
系 地 址 、 考 试 状态 和 考生 密码 等 属性 。 考 生 信息 实体 的 E-R 图 如 图 5.9 所 示 。 


图 5.9 考生 信息 实体 E-R 图 


2. 管理 员 实体 
管理 员 实 体 包括 编号 、 管 理 员 名 称 和 加 密 密 码 属 性 。 管 理 员 实体 的 E-R 图 如 图 5.10 所 示 。 


图 5.10 管理 员 实 体 E-R 图 


3. 考题 类 别 实体 
考题 类 别 实体 包括 编号 和 考题 类 别名 称 。 考 题 类 别 实体 的 E-R 图 如 图 5.11 所 示 。 
4. 考题 信息 实体 


考题 信息 实体 包括 编号 、 考 题 类 别 、 考 斌 成绩、 考题 内 容 、 考 题 答案 、 考 题 正 确 答案 及 考题 所 属 
套 题 等 属性 。 考 试题 目 实体 的 E-R 图 如 图 5.12 所 示 。 
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5.12 考题 信息 实体 E-R 


5.4.3 数据库 物理 结构 


根据 5.4.2 节 的 数据 库 概念 设计 ， 可 以 创建 与 实体 对 应 的 数据 表 。 为 了 使 读者 对 本 系统 的 数据 库 的 


结构 有 一 个 更 清晰 的 认识 , 下 面 给 出 数据 库 中 所 包含 的 Pn 

数据 表 的 结构 图 ， 如 图 5.13 所 示 。 [日 te 
本 系统 共 包 含 4 张 数据 表 ， 下 面 进行 详细 介绍 。 全 rad 
1. tb_user (考生 信息 表 ) L ”日 bt (考题 信息 表 ) 


考生 信息 表 用 来 保存 考生 信息 。 该 数据 表 的 结构 如 5.13 db_online 数据 库 所 包含 数据 表 的 结构 图 
图 5.14 所 示 。 

2. tb_admin (管理 员 信 息 表 ) 

管理 员 信息 表 用 来 保存 管理 员 的 用 户 名 和 密码 。 该 数据 表 的 结构 如 图 5.15 所 示 。 


tb_user 

online_id nt 4 v1 自动 编号 

nline nsw varchar |50 考生 姓名 

online_tel varchar 50 考生 联系 方式 

online_nunber varchar 50 考生 准 考证 号 

online_grads nt 4 V 考试 成 绩 > 

tb_admin 

online_subject varchar 50 i& 考题 类 别 | 

rr 户 和 日 v 考试 时 间 列 名 | 数据 类 型 | 长 度 [区 许 宝 | 标识 | 标识 种 子 | _ 撕 玉 习 
[| aine sdaress vc 100 考生 联系 地 址 li 目 动 篇 号 
加 im at a Iv 考试 你 态 区 2 管理 员 名 称 
| online pass varchar 50 考生 密码 rd rarchar 2 Vv 管理 员 密 码 

到 到 


图 5.14 考生 信息 表 结 构 图 5.15 管理 员 信 息 表 结 构 
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3. tb_ktlb (考题 类 别 信息 表 ) 

考题 类 别 信息 表 用 来 保存 考题 类 别 。 该 数据 表 的 结构 如 图 5.16 所 示 。 

4. tb_kt (考题 信 息 表 ) 

考题 信息 表 用 来 保存 考试 题目 信息 和 考题 答案 等 相关 信息 。 该 数据 表 的 结构 如 图 5.17 所 示 。 


t 
| 
varchar 50 考题 类 别 


E 列 名 | 数据 类 型 | 长 度 [允许 宝 | 标识 [标识 种 子 | 撕 玉 | 
an vi 目 动 编 叶 


‘online_ktlb warchar S50 考题 类 别 


到 


5.5.1 前 台 首页 概述 


考生 通过 “考生 登录 ”模块 的 验证 后 ， 可 以 登录 到 网 络 在 线 考试 系统 的 前 台 首 页 。 前 台 首 页 主要 
用 于 实现 前 台 功 能 导航 ， 该 页 面 主要 包括 考生 注册 、 考 生 登 录 、 修 改 密码 、 成 绩 查 询 、 进 入 考场 和 退 
出 信息 6 个 导航 链接 。 

下 面 看 一 下 本 案例 中 提供 的 前 台 首 页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\05\online_ks\index.php， 
如 图 5.18 所 示 。 


考生 注重 ”| 考生 登录 | 修改 密码 | 成 核查 淘 | 进入 考场 | 涝 出 信息 


图 5.18 网 络 在 线 考试 系统 前 台 首页 
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5.5.2 ”前台 首页 技术 分 析 


PHP 连接 SQL Server 数据 库 的 方法 有 两 种 : 一 是 通过 ADO 方式 连接 ; 二 是 通过 mssql_connect() 
函数 连接 。 下 面 分 别 介绍 这 两 种 连接 方式 。 
1. 通过 ADO 方式 连接 SQL Server 数据 库 


利用 ADO 方式 建立 与 SQL Server 数据 库 的 连接 ， 与 使 用 ADO 方式 建立 与 Access 数据 库 连接 的 
方法 类 似 ， 建 立 连接 是 在 本 实例 的 conn.php 文件 中 实现 的 。 其 代码 如 下 : 

<?php 

$conn=new com("adodb.connection"): 

$connstr="provider=sqloledb:data source=localhost:uid=sa:pwd=:database=db online"; 


$conn->open($connstr); 
?> 


2. 通过 mssql_connect() 函 数 连接 SQL Server 数据 库 
通过 mssql_connect() 函 数 建立 与 SQL Server 数据 库 的 连接 首先 应 该 配置 php.ini 文件 , 将 extension 


=php_mssql.dll 前 面 的 注释 去 掉 ， 然 后 重新 启动 Apache 服务 器 。 通 过 mssql_connect0 函 数 建立 与 数据 
库 连 接 的 代码 如 下 : 


<?php 
@ $conm=mssql connect("localhost"."sa"""]: // 建 立 与 SQL Server 数 据 库 的 连接 
四 mssql select_ db("db_online ",$conn): /选择 数据 库 

?> 
Ah 代码 贴 二 


@ mssql_connect() 浮 数 : 打开 一 个 到 SQL Server 服务 器 的 连接 。 如 果 成 功 ， 则 返回 一 个 SQL Server 连接 标识 ; 失 
败 则 返回 false。 


Tesource mssql_ connect ( [string servemame [, string username [, string password [, bool new_link]]]] ) 


参数 servemame 指 SQL Server 服务 器 ; 参数 usernamer 指 用 户 名 ; 参数 password 指 用 户 密码 ; 参数 new_link 指 连 
接 标识 。 

@ mssql select db0 函 数 : 选择 SQL Server 数据 库 。 如 果 成 功 ， 则 返回 true; 失败 则 返回 false。 

bool mssql_select db( string database_name [.resource link identifier]) 


参数 database_name 指数 据 库 名 称 ; 参数 link identifier 指 连接 标识 。 
5.5.3 前台 首页 的 实现 过 程 


网 络 在 线 考试 系统 前 台 首 页 主要 实现 了 考生 登录 功能 ， 考 生 通 过 准 考证 号 和 密码 进行 登录 。 
考生 登录 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 5.1 所 示 。 
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表 5.1 考生 登录 页 面 所 涉及 的 重要 表单 元 素 


名 称 


forml form 


重要 属性 
method="post" action="denglu okphp” | 表单 
id="online number" size="20" 


id="online pass" size="20" 


online number | text 


考生 在 考生 登录 页 面 录入 准确 的 准 考证 号 和 密码 后 ， 单 击 “ 提 交 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 
理 页 , 应 用 mssql_query0 函 数 检索 考生 是 否 成 功 登录 。 如 果 查 询 结果 为 真 , 则 将 准 考 证 号 存储 在 session 
变量 中 ， 否 则 弹出 提示 信息 ， 代 码 如 下 : 


例 程 01 代码 位 置 ; 光盘 VTMWS\online kswdenglu_ okphp 


<?php 

include("conn/conn.php"): // 连 接 数 据 库 文件 
S$online number=$_POST[online number]: // 获 取 准 考证 号 
Sonline pass=$_ POST[online pass]: /获取 考 生 密码 


本 半 于 市 训 市 市 市 训 市 订 计 证 字 市 计 证 齐 计 本 证 让 下 市 训 束 训 吕 可 本 中 村 考 生 的 准 考证 号 和 密码 是 否 正确 六 让 李 本 让 亨 本 机 训 训 刘 机 训 市 市 市 训 本 市 市 本 刘 训 计 事 率 训 束 */ 


©@ $query=mssql query("select * from tb_user where online number='$online number and online pass='$online pass"); 
四 iftmssql num rows($query)>0){ // 如 果 查 询 结 果 为 真 
上 session_register(online_ number): // 将 准 考证 号 存储 在 session 变 量 中 
echo "<script> alert(' 登 录 成 功 !): window.location.href='index.php?online= 考 试 规则 ';</script>"; 
jelse{ /如 果 没 有 检索 到 数据 
echo "登录 失败 1": // 弹 出 提示 信息 
?> 
< 代码 贴 十 


@ mssql query: 向 与 指定 的 SQL Server 服务 器 中 的 当前 活动 数据 库 发 送 一 条 SQL 查询 ， 语 法 如 下 : 
Tesource mssql query ( string query [, resource link identifier] ) 


参数 query 为 传 入 的 是 SQL 的 指令 ; 参数 link identifier 传 入 的 是 由 mssql connect(O 函 数 或 mssql_pconnect0) 函 数 返 
回 的 连接 号 ， 如 果 省 略 该 参数 ， 则 会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连接 。 

@ mssql num rows: 获取 结果 集中 行 的 数目 。 值 得 注意 的 是 ， 该 函数 仅 对 SELECT 语句 有 效 。 

上 @ session register: 在 整个 域 范围 内 增加 一 个 session 变量 ， 语 法 如 下 : 


boolean session register(string name) 


参数 name 用 于 指定 新 session 变量 的 名 称 。 
5.6 考生 信息 模块 设计 


5.6.1 考生 信息 模块 概述 


考生 信息 模块 主要 包括 考生 注册 、 考 生 登 录 、 修 改 密码 3 个 功能 。 考 生 首先 要 注册 为 网 站 用 户 ， 
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然后 才 被 授权 登录 网 站 进行 一 系列 操作 的 权限 ; 登录 后 考生 还 可 以 修改 个 人 的 密码 。 考 生 信息 模块 的 
系统 流程 如 图 5.19 所 示 。 


图 5.19 考生 信息 模块 的 系统 流程 图 


5.6.2 考生 信息 模块 的 技术 分 析 


在 考生 注册 信息 模块 中 应 用 到 Ajax 无 刷新 技术 获取 考生 的 准 考证 号 码 和 密码 。 在 介绍 该 功能 的 实 
现 过 程 之 前 ， 先 对 Ajax 技术 中 的 XmlHttpRequest 对 象 的 属性 和 方法 进行 详细 的 讲解 。 
XmlHttpRequest 对 象 是 Ajax 技术 的 核心 ， 有 关 该 对 象 的 属性 和 方法 的 详细 介绍 如 表 5.2 和 表 5.3 
所 示 。 
表 5.2 XmlHttpRequest 对 象 的 属性 


属 性 描述 
readyState 返回 当前 的 请 求 状态 
onreadystatechange 当 readyState 属性 改变 时 就 可 以 读 取 此 属性 值 
status 返回 Http 状态 码 
responseText 将 返回 的 响应 信息 用 字符 串 表示 
ResponseBody 返回 响应 信息 正文 ， 格 式 为 字 节 数组 
ResponseXML 将 响应 的 domcoment 对 象 解析 成 XML 文档 并 返回 
表 5.3 XmlHttpRequest 的 方法 
方 ” 法 描述 
Open 初始 化 一 个 新 请 求 
Send 发 送 请求 
GetAllReponseHeaders 返回 所 有 Http 头 信 息 
GetResponseHearder 返回 指定 的 Http 头 信息 
SetRequestHeader 添加 指定 的 Http 头 信息 
Abort 停止 当前 的 Http 请 求 
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下 面 对 XmlHttpRequest 对 象 的 常用 属性 和 方法 进行 详细 介绍 。 

1. readystate 属性 

readystate 属性 用 于 返回 当前 的 请 求 状态 ， 请 求 状态 共有 5 种 ， 如 表 5.4 所 示 。 
表 5.4 readystate 属性 值 


属 性 值 描 述 


0 | 表示 尚未 初始 化 ， 即 未 调用 open0 方 法 
1 | 建立 请 求 ， 但 还 未 调用 send0 方 法 发 送 请 求 
2 发 送 请 求 
3 处 理 请 求 
EN 
2. status 属性 


status 属性 用 于 返回 Http 状态 码 ， 常 用 的 Http 状态 码 如 表 5.5 所 示 。 
表 5.5 “Http 状态 码 


没有 发 现 文件 
服务 器 内 部 错误 
服务 器 不 支持 或 拒绝 请 求 中 指定 的 Http 版 本 


3. responseText 属性 

responseText 属性 将 返回 的 响应 信息 用 字符 串 来 表示 。 在 默认 情况 下 , 返回 的 响应 信息 的 编码 格式 
为 utf-8。 

4. responseXML 属性 

responseXML 属性 用 于 将 响应 的 domcoment 对 象 解析 成 XML 文档 并 返回 。 

5. open() 方 法 

open() 方 法 用 于 初始 化 一 个 新 的 请 求 。 

语法 : 

open(String method, String url, Boolean asyn, String user. String password) 
其 中 ，method 和 url 是 必 选 参数 ，asyn、user 和 password 是 可 选 参数 。open() 方 法 各 参数 如 表 5.6 所 示 。 

表 5.6 open 方法 参数 


参数 名 称 描 述 
method 此 参数 指明 了 新 请 求 的 调用 方法 ， 其 取 值 有 get 和 post 


ul 表示 要 请 求 页 面 的 ul 地 址 。 格 式 可 以 是 相对 路 径 、 绝 对 路 径 或 网 络 路 径 
as 说 明 该 请 求 是 异步 传输 还 是 同步 传输 ， 默 认 值 为 tue〈 允 许 异步 传输 ) 


User 服务 器 验证 时 的 用 户 名 
password 服务 器 验证 时 的 密码 
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6. send() 方 法 

send() 方 法 用 于 发 送 请 求 到 服务 器 。 

语法 : 

send(body) 

如 果 没 有 要 发 送 的 内 容 ， 则 body 可 以 省 略 或 为 Null。 

7. GetAlIReponseHeaders() 方 法 

GetAlIReponseHeaders() 方 法 用 于 获取 响应 的 所 有 Http 头 信息 。 获 取 到 的 信息 是 按照 “名 称 / 键 值 ” 
排列 的 ， 信 息 之 间 是 用 “: ”冒号 进行 分 隔 的 。 

语法 : 

GetAllReponseHeaders () 

8. GetResponseHeader() 方 法 

GetResponseHeader() 方 法 是 获取 响应 中 指定 的 Http 头 信息 。 

语法 : 

GetResponseHeader(String head) 

9. SetRequestHeader() 方 法 

SetRequestHeader() 方 法 用 于 添加 一 个 Http 头 信 息 。 

语法 : 

SetRequestHeader(name,value) 

其 中 ，name 表示 Http 头 名称 ，value 表示 信息 内 容 。 

10. Abort() 方 法 

Abort() 方 法 用 于 取消 一 个 请 求 。 

语法 : 

Abort0 


5.6.3 ”考生 注册 的 实现 过 程 


国 ”考生 注册 使 用 的 数据 表 : tb_user 

在 网 络 在 线 考试 系统 首页 ， 单 击 “ 考 生 注 册 ” 超 链接 ， 即 可 进入 考生 注册 页 面 ， 输 入 个 人 的 真实 
资料 后 ， 单 击 “ 注 册 ” 按 钮 ， 系 统 将 根据 输入 的 个 人 资料 自动 生成 准 考证 号 和 考生 密码 ， 如 图 5.20 
所 示 。 


[0] 说 明 : 考生 需要 通过 准 考证 号 和 密码 登录 网 络 在 线 考试 的 核心 模块 ， 即 进行 在 线 考试 及 成 绩 查询 
等 操作 。 
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考生 注册 | 考生 和 登录 1 能 改 密码 | 成 病 查 询 | 进入 考场 | 亡 出 竺 让 


| 


EJ 


小 可 必用 户 注 忆 磺 从， 居 昨 整 的 民 考 证 纯 克 982715. 和 路 克 276107. 


图 5.20 考生 登录 页 面 的 运行 结果 


在 考生 注册 页 面 zhuce.php 页 中 实现 考生 注册 信息 的 提交 ， 并 且 通 过 Ajax 的 页 面 无 刷新 技术 获取 
考生 的 准 考证 号 和 密码 。zhuce.php 页 中 的 关键 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 \TMVWS\online ksvzhuce .php 

<!--- 调 用 一 个 javascript 脚 本 文件 --> 

‘<script type="text/javascript" src="js/zhuce js.js"></script> 

<!--- 通 过 onclick 事 件 调用 一 个 自 定义 函数 --> 

<input type="button" name="Submit" value=" 注 册 " onClick="process()"> 

<!--- 输 出 一 个 div 标 签 中 的 值 -> 

<div id="divMessage" /> 

在 zhuce jsjs 文件 中 首先 定义 一 个 createXmlHttpRequestObject0 对 象 ， 并 获取 XMLHttpRrequest 
对 象 ， 然 后 定义 xmlHttp 用 来 存储 将 要 使 用 的 XMLHttpRrequest 对 象 。 关 键 代码 如 下 : 


例 程 03 ”代码 位 置 : 光盘 \TM\05\online_ks\js\zhuce jsjs 


var xmlHttp=createXmlHttpRequestObject(): /定义 XMLHttpRrequest 对 象 
function createXmlHttpRequestObjectO{ /获取 XMLHttpRrequest 对 象 
var xmlHttp: /用 来 存储 将 要 使 用 的 XMLHttpRrequest 对 象 
这 window.ActiveXObjecb{ // 如 果 在 internet Explorer 下 运行 
ty{ 
xmlHttp=new ActiveXObject("Microsoft. XMLHTTP"): 
}catch(e){ 
xmlHttp=false: 
} 
}else{ // 如 果 在 Mozilla 或 其 他 的 浏览 器 下 运行 
ty{ 
xmlHttp=new XMLHttpRequestO: 
}catch(e){ 
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XmlHttp=false: 


这 kxmlHttp) // 返 回 创建 的 对 象 或 显示 错误 信息 
alert(" 返 回 创建 的 对 象 或 显示 错误 信息 "); 
else 
Teturn xmlHttp: 
} 


然后 使 用 XMLHttpRequest 对 象 创建 异步 Http 请 求 ， 定 义 函 数 process0， 对 表单 中 提交 的 数据 进 
行 判断 ， 并 且 获 取 表 单 中 输入 的 信息 ， 在 服务 器 端 执行 zhuce_ok.php 文件 ， 向 服务 器 发 送 请 求 。 关 键 
代码 如 下 : 

例 程 04 ”代码 位 置 光盘 \TMVWOS\online ksWjsvzhuce jsjs 


function processO{ // 使 用 XMLHttpRequest 对 象 创建 异步 HTTP 请 求 

if(forml.usemame.value—""){ // 如 果 考 生 姓名 为 空 

alert(" 请 输入 姓名 !"): // 则 弹出 提示 
forml.username.select(): // 返 回 焦点 到 考生 姓名 文本 框 
return(false); 

} 

if(forml.tel.value==""){ // 如 果 考 生 联系 电话 为 空 
alert(" 请 输入 电话 号 码 ! "); // 则 弹出 提示 
form!l .tel.select|: // 返 回 焦点 到 考生 联系 方式 文本 框 
return(false): 
b 

if(checkphone(form!1 .tel.value)!=true){ // 如 果 考 生 联系 方式 格式 不 正确 
alert(" 您 输入 的 电话 号 码 的 格式 不 正确 !"); // 则 弹出 提示 
forml .tel.selectO: // 返 回 焦点 到 考生 联系 方式 文本 框 
return(false); 
b 

if(form1.address.value—""){ // 如 果 考 生 联系 地 址 为 空 
alert(" 请 输入 联系 地 址 ! "); // 则 弹出 提示 
forml.address.select0: // 返 回 焦点 到 考生 联系 地 址 文本 框 
Teturn(false): 


if(xmlHttp.readyState 一 4 || xmlHttp readyState==-0){ 。”// 在 xmlHttp 对 象 不 忙 时 进行 处 理 
names = document.getElementById("usemame").value: ”// 获 取 用 户 在 线 表单 中 输入 的 姓名 
tels = document.getElementById("tel").value: // 获 取 用 户 在 线 表单 中 输入 的 电话 
addresss =document.getElementById("address").value: 。”// 获 取 用 户 在 线 表 单 中 输入 的 地 址 
xmlHttp.open("GET","zhuce_ok.php?online User="+names+"& online tel="+tels+"& 

‘online address="+addresss.true); // 在 服务 器 端 执 行 zhuce_ok.php 
xmlHttp.onreadystatechange=handleServerResponse: 。“”// 定 义 获取 服务 器 端 响应 的 方法 


xmlHttp.send(null): // 向 服务 器 发 送 请 求 
}else 
setTimeout('process()'.1000): // 如 果 服 务 器 忙 ，1 秒 后 重 试 


. 
最 后 获取 从 服务 器 端 返回 的 消息 。 关 键 代码 如 下 : 
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例 程 05 ”代码 位 置 : 光盘 \TM\05\online_ ksijs\zhuce jsjs 


// 当 收 到 服务 器 端的 消息 时 自动 执行 
function handleServerResponseO{ 
这 xmlHttp .readystate—4){ /在 处 理 结束 时 进入 下 一 步 
if(xmlHttp.status—200){ // 状 态 为 200 表 示 处 理 成 功 结束 
xmlResponse=xmlHttp.responseXML:; // 获 取 服 务 器 端 发 来 的 XML 信 息 


xmlDocumentElement=xmlResponse.documentElement: ”// 获 取 XML 中 的 文档 对 象 ( 根 对 象 ) 
helloMessage=xmlDocumentElement.firstChild.data: // 获 取 第 1 个 文档 子 元 素 的 文本 信息 
// 使 用 从 服务 器 端 发 来 的 消息 更 新 客户 端 显示 的 内 容 
document.getElementById("divMessage").innerHTML='<i>'+helloMessaget+'</i>"; 
setTimeout('process()'.1000):; // 重 新 开始 

jelse{ 
// 如 果 Http 的 状态 不 是 200 表 示 发 生 错 误 
alert("There was a problem accessing the server:"+xmlHttp.statusText); 


) 


zhuce_ok.php 文件 在 服务 器 端 被 执行 ， 将 获取 到 的 用 户 信息 和 随机 生成 的 准 考证 号 、 密 码 添加 到 
数据 库 中 ， 并 且 将 准 考证 号 和 密码 返回 到 考生 注册 页 面 中 。 关 键 代 码 如 下 : 


例 程 06 ”代码 位 置 ， 光盘 \TM\05\online_ks\zhuce_ok.php 


<?php 
header('Content-Type: text/xml): // 创 建 一 个 XML 格 式 输出 
echo '<?xml version="1.0" encoding="gb2312" standalone="yes" ?>': // 创 建 XML 头 
echo '<response>'; // 创 建 <response> 元 素 

@ echo$online user=$ GET[online user]: // 获 取 考生 姓名 
S$online_tel=$_GET[online_tel]: // 获 取 考生 联系 电话 
Sonline address=$ GET[online address]: // 获 取 考生 联系 地 址 

四 S$online number=substr(mt_rand(100000.999999).0.6): // 生 成 6 位 随机 的 准 考证 号 码 
$online_ pass=substr(mt_rand(100000.999999).0.6): // 生 成 6 位 随机 的 考生 密码 


人 中 束 中 中 中 中 中 审 站 字 虽 中 中 下 中 中 中 中 中 中 让 可 中 中 下 中 中 站 所 从 客户 端 获取 的 用 户 创建 输出 于 六 六 六 六 闵 末末 冰冰 六 六 冰冰 来 六 玉 宁 六 末末 六 玉米 冰 玉 玉 站 冰冰 站 be 
include("conn/conn.php"); 
$query=mssql query("insert into tb_user(online_user,online tel,online_address,online number.online pass) 
values('$online_user','$online tel','$online_ address','$online number,Sonline pass)"): 
if($query—true){ 
echo $online_user: 


echo "用 户 注册 成 功 ， 这 是 您 的 准 考证 号 码 $online_number. 和 密码 $online_pass.": 


} 
echo '</response>"; // 关 闭 <response> 元 素 
?> 
< 代码 贴 十 
@ $_GET 预定 义 变量 : PHP 使 用 $_GET 预定 义 变量 自动 保存 通过 get 方法 传 过 来 的 值 ， 使 用 格式 为 : 
$_ GET[name] 


@ substr0 函 数 : 从 指定 的 字符 事 中 按照 指定 的 位 置 截 取 一 定 长 度 的 字符 ， 语 法 如 下 : 


string substr(string str.int start.int length) 
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参数 str 指定 字符 串 对 象 。 参 数 start 指定 开始 截取 字符 囊 的 位 置 。 如 果 参 数 start 为 负数 ， 则 从 字符 囊 的 末尾 开始 截 
取 。 参 数 length 是 可 选 参 数 ， 指 定 截取 字符 的 个 数 ， 如 果 length 为 负数 ， 则 表示 取 到 倒数 第 length 个 字符 。 


5.6.4 ”单元 测试 


在 应 用 Ajax 技术 实现 在 线 考试 的 用 户 注册 模块 后 ， 为 保证 程序 能 够 正常 运行 , 必须 对 其 进行 测试 。 
运行 考生 注 


全 TD 


图 5.21 考生 注册 模块 运行 出 现 的 错误 


从 弹出 的 错误 信息 来 看 ， 是 程序 中 与 JavaScript 脚本 有 关 的 内 容 出 现 问 题 , 经 过 反复 测试 和 查找 发 
现 ， 问 题 不 是 出 现在 zhuce_jsjjs 文件 中 ， 因 为 在 该 文件 中 定义 的 自 定义 函数 process0) 是 可 以 运行 的 ， 
所 以 问题 应 该 是 出 现在 对 服务 器 返回 的 信息 进行 输出 中 ， 即 zhuce_ok.php 文件 中 。 对 该 文件 进行 分 析 
发 现 ， 原 来 在 应 用 Ajax 技术 中 ， 服 务 器 端 返回 的 消息 采用 的 是 XML 格式 。 可 是 在 客户 端 接收 XML 
文档 时 却 忽略 了 “header(CContent-Type: text/xml);:” 头 部 信息 的 使 用 ， 从 而 导致 客户 端 用 于 解析 XML 
的 API 抛 出 一 个 错误 信息 。 只 有 在 设置 头 部 信息 (header(Content-Type: text/xml):) 后 ， 客 户 端 才 可 以 
正确 地 接收 到 服务 器 端 传递 的 XML 文档 信息 。 正 确 的 程序 代码 如 下 


<2php 
// 创 建 一 个 XMI 格 式 输出 


header('Content-Type: text/xml'); // 标 记 输 出 的 是 XML 文档 
/创建 XMLI 头 

echo '<?xml version="1.0" encoding="gb2312" standalone= "yes" ?>' 

>: 


5.7 在 线 考试 模块 设计 
5.7.1 在 线 考试 模块 的 概述 


在 线 考试 模块 的 主要 功能 是 允许 考生 在 网 站 上 针对 指定 的 课程 进行 考试 。 在 该 模块 中 ， 考 生 首先 
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需要 登录 到 本 系统 中 ， 阅 读 考 试 规则 ， 在 同意 所 列 出 的 考试 规则 后 ， 才 能 选择 考试 类 别 和 套 题 ， 然 后 
进入 考试 页 面 进行 答题 ， 当 考生 提交 试卷 或 者 到 达 考试 结束 时 间 时 ， 系 统 将 自动 对 考生 提交 的 试卷 进 
行 评分 ， 并 给 出 最 终 考试 成 绩 。 在 线 考试 模块 的 系统 流程 如 图 5.22 所 示 。 


| 选择 考题 类 别 和 套 题 | 


l 


提交 试卷 /到 达 考试 结束 时 间 


显示 本 次 考试 成 绩 


图 5.22 在 线 考试 模块 的 系统 流程 图 
5.7.2 ”在线 考试 模块 的 技术 分 析 


在 线 考试 模块 中 最 核心 的 功能 是 如 何 输出 考题 、 如 何 对 提交 的 答案 进行 判断 ， 并 且 将 答案 输出 到 
当前 页 中 ， 这 是 在 线 考试 最 关键 之 处 。 下 面 就 讲解 一 下 该 技术 的 实现 方法 ， 由 于 上 述 3 方面 的 内 容 都 
是 在 同一 页 面 中 显示 ， 所 以 在 技术 实现 过 程 中 这 3 个 方面 的 内 容 是 以 一 个 整体 形式 出 现 。 这 里 以 单 选 
题 为 例 进 行 讲解 。 

首先 以 上 一 页 表单 中 提交 的 值 为 条 件 , 执行 查询 语句 , 通过 while 循环 语句 输出 查询 结果 ; 然后 根 
据 输出 的 查询 结果 创建 单 选 按钮 组 ， 生 成 考题 ; 最 后 ， 将 考题 答案 提交 到 本 页 ， 对 提交 的 答案 与 数据 
库 中 存储 的 正确 答案 进行 比较 ， 并 且 将 答案 输出 到 本 页 中 。 单 选 题 输出 、 答 案 判 断 和 考题 答案 输出 的 
关键 代码 如 下 : 


例 程 07 ”代码 位 置 : 光盘 \VTMVWOSVonline ks\ksks php 
<2php 
上 根据 提交 的 变量 执行 查询 语句 ， 从 数据 库 中 读 取 数据 ， 指 定 考题 的 类 型 为 0， 表 示 为 单 选 题 */ 


$query0=mssql query("select * from tb_ kt where kt lb='$kt lbes' and kt Ix='"0' and kt small lb='$kt small 1b"™"): 
$x=1: // 定 义 变量 Sx=1， 用 于 输出 题 号 
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Sfen0=0; /定义 变量 Sfen0=0， 用 于 考试 分 数 的 计算 
while($myrow0=mssql fetch array($query0)){ // 执 行 while 循 环 语句 

> 

<tr> 


<!--- 输出 单 选 题 的 题 号 、 考 题 内 容 以 及 该 道 体 的 分 数 ---> 
<td width="443" height="20" bgcolor="#FFFFFF">énbsp:; énbsp:<?php echo $x.".". $myrowO["kt_ nr"]?> </td> 
<td bgcolor="#FFFFFF"><?php echo $myrowO[kt fs]:?></td> 


0 > 
</t> 
<?php 
/***#* 获 取 考 题 提交 的 答案 ， 将 提交 的 考题 答案 进行 分 割 ， 并 赋 给 变量 Sstr0*******/ 
Sarray0=explode("*",S$myrowO["kt_daan"]): // 将 数据 库 中 的 考题 答案 以 * 进 行 分 割 , 并 写 入 到 数组 中 
if($_POST[Submit]!=""){ /判断 表单 提交 是 否 为 空 
for($a=0:$a<count($array0):$at+){ 1/ 循环 输出 数组 中 的 答案 
if($arrayO[$a]!=""){ // 数 组 中 的 值 不 为 空 


这 Sarray0[$a] 一 $_ POST[Smyrow0[kt id]]) {// 如 果 数 组 中 的 值 与 表单 提交 的 值 相 等 
$str0=$ POST[SmyrowO[kt id]]: // 将 表单 提交 的 值 赋 给 变量 $str0 


} 
} 
} 
/于 来 冰 来 康 率 认 闪 六 闪闪 率 来 率 来 率 闪 认 闪闪 闪闪 闪 站 站 让 水 农 灾 兴 补 率 闪闪 闪闪 站 让 永 让 灾 冰 素来 来/ 
for($a=0:$a<count($array0):Sat+){ // 执 行 for 循 环 语句 ， 根 据 考题 答案 来 决定 循环 的 次 数 
if($arrayO[$a]!=""){ // 浏 断 如 果 考 题 答案 不 为 室 ， 则 执行 下 面 的 内 容 

?> 
<h> 


<td height="20" bgcolor="#FFFFFF">&nbsp:&nbsp: 
<!--- 创建 单 选 按钮 组 ， 用 于 提交 答案 ---> 
<input type="radio" name="<?php echo $myrowO[kt id]:?>" value="<?php echo $array0[$a]:?>"> 


<?php echo $array0[$a]:?> 
<!----- 一 -一 -一 -一 --- 一 -一 -一 --------: > 
</td> 
<td bgcolor="#FFFFFF"> &nbsp: 
<? 
7 oe 断 提 交 的 答案 是 否 正 确 并 并 且 输 出 答 案 站 中 中 中 中 中 机 机 可 中 事 训 下 下 事 吕 中字 可 水 束 吕 事 吕 束 吕 be 
if($_POST[$myrowO[kt id]]==true){ // 判 断 表单 提交 的 值 是 否 为 空 
if($a—0) // 如 果 变 量 $a 一 0 则 输出 下 面 的 内 容 
if($myrowO["kt_zqdaan"]—$strO){ // 判 断 提 交 的 答案 是 否 与 数据 库 中 的 正确 答案 相同 
// 如 果 提 交 的 答案 正确 ， 则 输出 下 面 的 内 容 
echo "您 输入 的 答案 &nbsp:": 
echo "<font color=#FF0000'>".substr($str0.,0,1)."</font>"; // 获 取 答案 的 选项 
echo "&nbsp: 正 确 &nbsp:&nbsp: 分 数 :": 
echo "<font color=#FF0000'>". $myrowO[kt_fs]."</font>": /输出 考题 的 分 数 
Sfen0+=$myrowO["kt fs"]: // 对 考题 的 分 数 进行 合算 
}else{ // 如 果 提 交 的 答案 不 正确 则 输出 下 面 的 内 容 
echo "您 输入 的 答案 &nbsp:"; 
echo "<font color=#FF0000'>".substr($str0.0.1)."</font>"; 
echo "&nbsp: 错 误 &nbsp:&nbsp:": 
echo "正确 答案 :&nbsp:<font color='#FF0000'>".substr(SmyrowO[kt_zqdaan].0.1)."</font>" : 
b 
?></td> 
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} Sxtt; // 实 现 考 题 题 号 的 循环 


5.7.3 应 用 Ajax 在 线 答题 的 实现 过 程 


在 线 答 题 是 本 项 目 中 最 核心 的 技术 ， 也 是 本 系统 开发 的 最 终 目 的 。 在 线 答 题 模块 的 主要 功能 是 实 
现在 线 考试 ， 在 当前 页 面 中 输出 考题 答案 和 考试 成 绩 ， 并 且 采 用 计时 控制 考试 时 间 ， 如 果 考 试 超过 考 
试 时 间 ， 则 系统 自动 提交 考题 答案 ， 并 给 出 考试 成 绩 。 

考试 时 间 计 时 和 剩余 时 间 的 计算 是 应 用 Ajax 技术 来 实现 的 ， 有关 该 技术 的 实现 方法 可 以 参考 本 章 
开发 技巧 与 难点 分 析 中 的 通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 一 节 ， 这 里 不 讲解 。 

在 线 答 题 技术 中 对 单 选 题 的 处 理 方法 已 经 在 技术 分 析 中 进行 了 详细 的 讲解 ， 这 里 不 再 袭 述 。 下 面 
讲解 在 线 答题 中 多 选 题 的 处 理 方法 ， 其 实现 的 原理 与 单 选 题 是 相同 的 ， 只 是 在 输出 考题 的 答案 时 使 用 
的 是 复 选 框 ， 不 再 是 单 选 按钮 组 ， 并 且 在 对 复 选 框 名 称 进行 设置 时 增加 一 个 变量 来 对 不 同 考题 的 答案 
进行 区 分 。 首 先 以 上 一 页 表单 中 提交 的 值 为 条 件 ， 执 行 查询 语句 ， 通 过 while 语句 输出 查询 结果 ; 然后 
根据 输出 的 查询 结果 创建 复 选 框 ， 生 成 考题 ， 最 后 ， 将 考题 答案 提交 到 本 页 ， 对 提交 的 答案 与 数据 库 
中 存储 的 正确 答案 进行 比较 ， 并 且 将 答案 输出 到 本 页 中 。 

首先 从 数据 库 中 读 取 数据 ， 定 义 输出 考题 时 应 用 到 的 变量 值 ， 并 且 输 出 考题 的 编号 和 内 容 。 关 键 
代码 如 下 ; 

例 程 08 ”代码 位 置 ， 光盘 VTMVWOS\online_ks\ksksphp 

<?phl 

/根据 表单 提交 的 值 执行 查询 语句， 从 数据 库 中 读 取 符合 条 件 的 数据 

$query1=mssql_ query("select * from tb_kt where kt_lb='$kt lbes' and kt_lx='1' and kt small lb='$kt_small 1b"™); 


Sy=1; // 定 义 变量 $y 用 于 生成 考题 编号 
Sfen1=0: // 定 义 变量 $fen1 用 于 分 数 的 统计 
Sii=0; // 定 义 变量 $ii 用 于 控制 考题 答案 输出 的 次 数 
while(Smyrowl=mssql_fetch array($query1)){ /循环 输出 数据 
?> 
到 输出 考题 编号 和 考题 的 内 容 --- 一 一 -一 > 
<t> 


<td height="20" bgcolor="#FFFFFF">&nbsp:&nbsp:<?php echo $y.".".$myrow1["kt_nr"]?> </td> 
<td bgcolor="#FFFFFF"><?php echo Smyrowl[kt fs]:?></td> 
</t> 


< --- 一 ------ 一 . 一 一 一 -一 ~ > 


然后 ， 对 表单 中 提交 的 答案 进行 处 理 ， 与 数据 库 中 存储 的 正确 答案 相 比 较 ， 如 果 答案 正确 ， 则 将 
表单 提交 的 答案 写 入 到 一 个 空 数组 中 ， 代 码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\05\online ks\ksks php 


<2php 
提交 的 答案 进行 处 理 ， 输 出 考题 答案 + 
Sarray_al=array(): // 定 义 一 个 空 数组 


// 将 数据 库 中 考题 的 答案 以 * 进 行 分 割 ， 存 储 到 数组 Saray1 中 
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© Sarrayl=explode("*",.$myrow1["kt_daan"]): 
if($_ POST[Submit]!=""){ // 判 断 表单 提交 的 值 是 否 为 空 
for($i=0:$i<count($array1):$it+){ // 循 环 读 取 数 组 $array1 中 的 考题 答案 
if($array1[$i]!=""){ /独断 数 组 中 的 答案 是 否 为 空 


/对 表单 中 提交 的 答案 与 数据 库 中 的 答案 进行 比较 
if($array1[$i]=——=$_POST[$Smyrow1[kt id]."-".$i]) { 
// 如 果 答 案 相 同 则 将 表单 提交 的 答案 写 入 到 空 数组 Sarray al 中 
array_push($array al,$ POST[Smyrow1l[kt id]."-".$i]); 


< 人 代码 贴 二 
@ explode0 函 数 : 将 字符 串 依 指定 的 字符 串 或 字符 separator 分 开 。 
array explode(string separator, string string, [int limit]) 


返回 由 字符 串 组 成 的 数组 ,每 个 元 素 都 是 string 的 一 个 子囊 ,它们 被 字符 串 separator 作为 边界 点 分 隔 出 来 。 如 果 设 
置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 将 包含 string 的 剩余 部 分 ; 如 果 separator 为 空 字 
符 串 ("" )，explode0) 函 数 将 返回 false; 如 果 separator 所 包含 的 值 在 string 中 找 不 到 , 那么 explode0 函 数 将 返回 包含 string 
单个 元 素 的 数组 ;如果 参数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 

@ armray_ push0O 函 数 : 将 数组 当成 一 个 栈 ， 并 将 传 入 的 变量 压 入 该 数组 的 末尾 。 该 数组 的 长 度 将 增加 入 栈 变量 的 数 
目 。 返 回 数组 新 的 单元 总 数 。 


int array_push ( array array, mixed var [, mixed ...]) 


参数 array: 指 输入 的 数组 ; 参数 Var: 必要 参数 ， 用 来 压 入 数组 的 。 


接着 创建 考题 提交 的 复 选 框 ， 根 据 考题 的 ID 和 变量 $I 定义 复 选 框 的 名 称 ， 代 码 如 下 : 


例 程 10 ”代码 位 置 ， 光盘 \TM\05\online_ks\ksks.php 

<?php 
S$strl=implode("*".Sarray al): /以 * 来 分 割 数组 Sarray al 中 的 数据 ， 返 回 值 为 变量 $strl 
for($i=0:$i<count($array1):$i++){ /根据 数据 库 中 存储 的 答案 执行 循环 语句 


if($array1[$i]!=""){ / 浏 断 答案 不 为 空 
?> 


<tr><td height="20" bgcolor="#FFFFFF">&nbsp:&nbsp 


<- -一 一- 一 --- 创 建 答案 提交 的 复 选 框 ， 并 有 设置 复 选 杠 的 名 称 为 考题 的 id 和 变量 $i-------- 


<input type='checkbox' name='<?php echo Smyrow1l[kt id]."-".$i?>' value="<?php echo $array1[$i]:?>'> <?php echo 
Sarray1[$i]:?> 


</td><td bgcolor="#FFFFFF">é&nbsp:; 


再 次 对 提交 的 答案 进行 判断 ， 其 中 主要 应 用 的 是 explode()、list0 和 substr0 函 数 来 实现 答案 的 判断 
和 输出 。 程 序 代码 如 下 : 


例 程 11 代码 位 置 : 光盘 \TMVWSVonline ks\ksks.php 


?php 
/二 汪汪 闻 汪汪 汪汪 汪汪 于 * 汪 ** 对 提交 的 答案 进行 判断 ， 并 且 输 出 答案 和 最 终 得 分 关 兴 池 于 池 池 理 汪汪 业 池 字 池 汪汪 刘 浊 汪 汪 浊 人 
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这 $_POST[$myrowl[kt id]."-".3i]—true){ // 判 断 提交 的 答案 是 否 为 真 
if($ii—0){ // 判 断 变 量 $i 的 值 是 否 为 0， 控 制 答案 输出 的 次 数 
if($myrow1["kt zqdaan"]==$str1){ // 判 断 表单 提交 的 答案 是 否 与 数据 库 中 的 正确 答案 相等 
/本 本 示 机 本 本 机 本 本 本 本 本 本 本 本 如] 果 答案 正确 ， 则 输出 提交 的 答案 并 且 给 出 该 题 的 得 分 站 本 让 机 本 机 机 本 中 员 机 机 本 于 | 
echo "您 输入 的 答案 &nbsp:": 
Sarrayesg=explode("*",$str1): // 以 * 来 分 隔 提 交 的 答案 
© for($gg=0:$gg<count($arrayesg):$gg+H){ // 循 环 读 取 答 案 的 内 容 
@ list($Sname.$valuesg)=each(Sarrayesg): /输出 答案 


echo "<font color=#FF0000>".substr(S$valuesg.0.1)."</font>": /截取 答案 的 内 容 


} 
echo "&nbsp: 正 确 &nbsp:&nbsp: 分 数 :": 
echo "<font color=#FF0000'>". $myrow1[kt fs]."</font>"; 


Sfenl+=$myrow1["kt fs"]: /统计 考题 的 分 数 

}else{ 

/* * 直 办 * 相 站 # 加 果 输 出 的 答案 错误 ， 则 提示 您 输入 的 答案 错误 ， 并 给 出 正确 的 答案 *****t 4/ 
echo "您 输入 的 答案 &nbsp:"; 


Sarrayes=explode("*",$str1); 
for($g=0;$g<count($arrayes):Sg++){ 
list($name,$values)=each($arrayes): 
echo "<font color=#FF0000'>".substr($values,0,1)."</font>"; 


! 
echo "&nbsp: 错 误 &nbsp:&nbsp: 正 确 答案 :&nbsp:": 
Sarrayes=explode("*",$myrow1[kt_zqdaan]): 
for($g=0:$g<count($amayes):$g++H){ 
list($name,.$values)=each($arrayes): 
echo "<font color='#FF0000>".substr($values,0.1)."</font>": 


上 
}S$ii=1: // 控 制 每 道 题 的 答案 只 输出 一 次 
?></td></tr> 

<?php } } 

S$y++: // 控 制 每 道 题 的 题 号 

} 
?> 

外 代码 贴 十 


@ countO 函 数 : 返回 数组 中 的 单元 数目 ， 用 来 计算 数组 中 值 的 个 数 。 
int count ( mixed array [. int mode]) 
参数 array: 必要 参数 ， 输 入 的 数组 ; 参数 mode: 可 选 参数 ， 此 参数 的 默认 值 是 0。 
@ list0 函 数 : 把 数组 中 的 值 赋 给 一 些 变 量 . 与 arrayO 函 数 类 似 ， 这 不 是 真正 的 函数 ， 而 是 语言 结构 。list0 函 数 仅 能 
用 于 数字 索引 的 数组 ， 且 数字 索引 从 0 开始 。 
void list ( mixed ...) 
参数 mixed 为 被 赋值 的 变量 名 称 。 
each0 函 数 : 返回 数组 中 当前 指针 位 置 的 键 名 和 对 应 的 值 ， 并 向 前 移动 数组 指针 . 
array each ( array array) 


参数 array 为 输入 的 数组 。 
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5.7.4 “分 数 统计 和 成 绩 保 存 的 实现 过 程 


在 线 考试 模块 中 不 但 完成 考试 的 答题 和 判 卷 的 操作 ， 而 且 可 以 直接 对 考试 的 分 数 进行 统计 ， 获 取 
考生 的 最 终 得 分 ， 将 考生 的 成 绩 保 存 到 数据 库 中 ， 并 且 更 改 数据 库 中 考生 的 考试 信息 ， 说 明 该 考生 已 
经 完成 本 次 考试 ， 不 可 以 再 进行 本 类 别 的 考试 。 考 试 分 数 统计 和 存储 考试 成 绩 的 关键 代码 如 下 : 

例 程 12 ”代码 位 置 光盘 \TMVWOS\online ks\ksks .php 


<?php 
/让 二 机 于 可 本 事主 站 训 本 站 中 可 让 语 # 顷 计 -考生 的 考试 成 绩 ， 其 中 的 变量 分 别 代表 考生 各 个 题 型 的 得 分 二 # 本 让 训 本本 让 让 本 下 
S$zf=$fen0+$fenl+$fen2+$fen3; // 考 试题 分 数 统计 
echo "您 的 总 成 绩 是 :": /输出 字符 串 
echo $2zf: // 输 出 考试 成 绩 
?> 
< 
/本 宙 市 计 囊 训 事 刘 刺 计 字 字 市 计 认 计 亨 本 计 让 市 训 训 训 市 本 本 机 本 主 本 | 告 老 生 的 程序 存储 到 数据 库 中 闻 李 李 让 本本 机 可 本 证 市 事 语 训 刘 下 让 市 机 市 宙 训 训 刘 训 率 训 事 训 于 事 训 */ 
if($Submit==" 提 交 "){ // 首 先 判 断 考生 是 否 提交 考题 的 答案 
Sdata=date("Y-m-d H:i:s"); // 获 取 考试 的 时 间 


/二 这 半球 率 六 素 这 兴 闪 站 守 汪 本 守 站 中 村 丰 相 站 村 检 相 必 老生 的 程序 存储 到 数据 库 中 于 下 于 玉 玉 玉 永 冰 素 素 于 六 素 素 六 素 六 六 这 冰冰 站 / 

$grade="update tb_user set online grade='$zf,online subject='$ POST[kt lbes]',online pt="1',online date='$data' where 
online number='$_SESSION[online number]"; 

S$erade result=mssql query($grade): 

1 


?> 
5.7.5 单元 测试 
在 完成 在 线 考试 模块 的 设计 后 对 该 模块 进行 测试 ， 发 现在 执行 多 选 题 提 交 的 过 程 中 ， 没 有 任何 值 


返回 。 但 是 在 执行 单 选 题 提交 时 可 以 正常 运行 ， 查 看 该 文件 的 源 文件 发 现 ， 原 来 错误 出 现在 复 选 框 名 
称 的 设置 上 ， 所 有 复 选 框 都 是 相同 的 值 ， 其 运行 结果 如 图 5.23 所 示 。 


<tr> 
<td height="28" bgcolor="NFFFFFF">8nbsp;énbsp; 
<input type='checkbox” nane='17° value="C PHP*> C PHP</td> 
《td bgcolor="WFFFFFF">&nbsp;</td> 


/tr> 
<tr> 
<td “20" bgcolor='"" 抽 FFFFF">&nbspi&nbsp， 
<input type= “checkbox”mane='17 oaluesip asp> D aspe/te> 
《td bgcolor="WFFFFFF">&nbsp;</td> 
/tr> 


图 5.23 查看 源 文件 的 结果 


在 设置 复 选 框 的 名 称 时 必须 为 每 一 个 题 设 置 不 同 的 复 选 框 名 称 ， 否 则 将 不 能 获取 到 表单 提交 的 数 
据 ， 这 就 是 多 选 题 与 单 选 题 进行 输出 和 判断 的 主要 区 别 ， 多 选 题 中 有 多 个 答案 ， 而 单 选 题 只 有 一 个 
所 以 单 选 题 的 单 选 按 钮 组 的 名 称 可 以 直接 使 用 考题 的 ID 进行 命名 ， 但 是 多 选 题 必须 在 考题 ID 的 基础 
之 上 在 定义 一 个 变量 来 区 分 每 个 复 选 框 的 名 称 。 在 程序 中 ， 错 误 的 复 选 框 名 称 定义 的 代码 如 下 : 
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<input type="radio" name="<?php echo SmyrowO[kt id]:?>" value="<?php echo SarrayO[$a]:?>"> 
正确 的 复 选 框 名 称 定义 应 该 是 在 考题 ID 的 基础 之 上 加 一 个 变量 31， 程 序 代码 如 下 : 
<input type='checkbox' name="<?php echo Smyrow1[kt_id]."-".$i?>' value='<?php echo Sarray1[$i]:?>> 
对 程序 代码 进行 修改 ， 重 新 保存 ， 程 序 可 以 正常 运行 。 


5.8 后 台 首 页 设计 


5.8.1 后 台 首页 概述 


网 络 在 线 考试 系统 的 后 台 首 页 是 管理 员 对 网 站 信息 进行 管理 的 首页 面 。 在 该 页 面 中 ， 管 理 员 可 以 
清楚 地 了 解 网 站 后 台 管理 系统 包含 的 基本 操作 。 网 络 在 线 考试 网 后 人 台 首 页 包含 的 主要 模块 如 下 : 
管理 员 信息 管理 : 主要 用 于 修改 管理 员 信息 。 
考生 信息 管理 ， 主 要 包括 查看 注册 考生 信息 列表 和 考生 信息 查询 、 考 试 成 绩 查询 和 删除 已 注 
册 的 考生 信息 。 
考题 类 别管 理 : 主要 包括 查看 考题 类 别 列表 、 添 加 考题 类 别 信息 和 删除 考题 类 别 信息 。 
考题 信息 添加 : 主要 用 于 添加 为 各 类 套 题 添加 单 选 题 、 多 选 题 、 问 答题 和 论述 题 ， 并 设置 每 
题 的 分 数 及 内 容 。 
考题 信息 管理 : 主要 包括 查看 考题 类 别 列表 、 修 改 套 题 信息 和 删除 套 题 信息 。 
包括 查看 考试 题目 列表 、 添 加 考试 题目 、 修 改 考 试题 目 和 删除 考试 题目 。 
退出 管理 : 主要 用 于 退出 后 台 管 理 系统 。 

下 面 看 一 下 本 案例 中 提供 的 后 人 台 首 页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\05\online_ks\admin\ 
index.php， 如 图 5.24 所 示 。 


辐 回 


网 加 


办 多 办 


全 在 由 者 斌 六 后 人 管理 ;考生 信和 管理 


its 


考生 电话 
Ma3I-84972261 
Da31-44972266 


5.24 ”网 络 在 线 考 试 系统 的 后 台 首页 
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5.8.2 后台 首页 技术 分 析 


网 络 在 线 考 试 后 台 应 用 switch 语句 实现 的 一 种 类 似 于 框架 的 网 页 嵌 套 技术 。 为 了 能 够 更 好 地 理解 
这 个 技术 ， 下 面 进行 详细 介绍 。 

switch 语句 一 行 接 一 行 地 执行 (实际 上 是 语句 接 语 句 ) ， 开 始 时 没有 代码 被 执行 。 仅 当 一 个 case 语 
句 中 的 值 和 switch 表达 式 的 值 匹 配 时 ，PHP 才 开 始 执行 语句 ， 直 到 switch 的 程序 段 结束 或 者 遇 到 第 一 个 
break 语句 为 止 。 如 果 不 在 case 的 语句 段 最 后 写 上 break 的 话 ，PHP 将 继续 执行 下 一 个 case 中 的 语句 段 。 

switch 语句 的 语法 如 下 : 

switch(expr) { 


case value : statement' break: 


default : statement: break; 


} 
switch 语句 的 参数 说 明 如 表 5.7 所 示 。 
表 5.7 ”switch 语句 的 参数 说 明 


必要 参数 。 定 义 判断 的 条 件 表 达 式 
statement 必要 参数 。 满 足 条 件 后 ， 循 环 执行 的 语句 


5.8.3 后台 首页 的 实现 过 程 


网 络 在 线 考试 后 台 主 要 应 用 了 switch 语句 和 include 包含 语句 实现 了 类 似 于 框架 的 网 页 典 套 技术 。 
其 中 ， 应 用 switch 语句 ， 根 据 变量 标识 $htgl 提交 的 值 进行 判断 ， 应 用 include 包含 语句 调用 不 同 功能 
模块 的 脚本 文件 ， 代 码 如 下 : 

例 程 13 ”代码 位 置 : 光盘 \TM\05\online_ks\admin\index.php 


<?php 
switch($htgl) { // 判 断后 台 超 链 接 名 称 
case "考生 信息 管理 ": // 如 果 名 称 为 “考生 信息 管理 ” 
include("ksxx_gl.php"): // 则 应 用 includeO 语 名 调用 ksxx_gl.php 文 件 
break: // 退 出 语句 段 的 执行 
case "考题 类 别管 理 ": // 如 果 名 称 为 “考题 类 别管 理 ” 
include("ktlb_gl.php"): // 则 应 用 includeO 语 句 调用 ktlb_gl.php 文 件 
break: // 退 出 语句 段 的 执行 
case "考题 信息 添加 ": // 如 果 名 称 为 “考题 信息 添加 ” 
include("ktxx_tj.php"): // 则 应 用 includeO 语 句 调用 ktxx_tj.php 文 件 
break: // 退 出 语句 段 的 执行 
case "考题 信息 管理 ": // 如 果 名 称 为 “考题 信息 管理 ” 
include("ktxx_gl.php"): // 则 应 用 includeO 语 句 调用 ktxx_gl.php 文 件 
break: // 退 出 语句 段 的 执行 
} 
?> 
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5.9 ”考题 信息 管理 模块 设计 


5.9.1 ”考题 信息 管理 模块 的 概述 


考题 信息 管理 模块 主要 包括 查询 考题 信息 、 添 加 考题 信息 、 修 改 考题 信息 和 删除 考题 信息 等 4 个 
功能 。 考 题 信息 管理 模块 的 框架 如 图 5.25 所 示 。 


Tawiens [| [woe [| [wavens [|[] wv | 


图 5.25 考题 信息 管理 模块 的 框 染 图 
5.9.2 ”考题 信息 管理 模块 的 技术 分 析 


在 实现 考题 信息 管理 模块 时 ， 为 了 更 好 地 管理 ， 因 此 把 考题 类 别 单独 存储 于 一 个 数据 表 tb_lb 中 ， 
这 样 在 录入 考题 信息 时 就 可 以 把 考题 类 别 以 下 拉 列 表 的 形式 从 数据 库 中 读 取出 来 。 这 种 从 下 拉 列 表 中 
动态 显示 数据 表 某 列 的 字段 值 的 方法 ， 不 仅 方便 管理 员 更 快捷 、 灵 活 地 操作 网 络 在 线 考试 系统 ， 更 大 
大 提高 了 工作 效率 ， 达 到 事半功倍 的 效果 。 

下 拉 列 表 是 一 种 最 节省 空间 的 数据 显示 方式 ， 正 常 状态 下 只 能 看 到 一 个 选项 ， 单 击 控制 按钮 后 ， 
可 以 显示 一 定数 量 的 选项 ， 如 果 超 出 这 个 数量 ， 会 自动 显示 滚动 条 ， 管 理 员 可 以 通过 拖 动 滚动 条 来 选 
择 各 选项 。 

下 面 介绍 从 下 拉 列 表 中 动态 显示 数据 表 某 列 的 字段 值 的 方法 。 

首先 ， 创 建 与 数据 库 的 连接 ， 代 码 如 下 : 


<?php 
S$id=mssql_connect('localhost','sa',”): 
mssql_select_db("db_online ".$id): 
?> 


然后 应 用 下 拉 列 表 框 和 select 查询 语句 相 结合 实现 在 下 拉 列 表 中 显示 数据 表 中 的 online ktlb 字段 
的 值 ， 通 过 while 循环 语句 进行 输出 ， 代 码 如 下 : 


<?php 

$query=mssql query("select * from tb_ktlb"): // 查 询 考题 类 别 表 中 的 信息 
while($Smyrow=mssql fetch array($query)){ // 应 用 while 循 环 语句 输出 考题 信息 
?> 

< 一---------- 输 出 考题 类 别 表 中 的 online ktib 字 段 -一 一 一 一 一 一 一 一 一 一 一 -一 
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<option value="<?php echo $myrow[online ktlb]:?>"><?php echo $myrow[online ktlb]:?></option> 
<?php 


上 //while 循 环 语句 结束 
?> 


5.9.3 ”考题 信息 添加 的 实现 过 程 


国 ”添加 考试 题目 信息 使 用 的 数据 表 : tb Lesson、tb_taoTi 和 tb_Questions 


考试 题目 添加 包含 两 个 步骤 : 一 是 为 添加 的 考试 题目 选择 专业 、 课 程 以 及 套 题 ， 二 是 将 填写 的 考 
试题 目 信息 插入 到 数据 库 中 。 考 题 信息 添加 页 面 的 运行 结果 如 图 5.26 所 示 。 


会 在 由 忆 二 不 较 后 各 入 理 ;考题 信息 沫 加 


孝 雪 |[F 悦 | [而 融 到 。 | 雪夫 [玉生 可。 [4 数 鞠  ， 
两 字 司 ， 个 函数 ?了 了 


习 


tin GEEE GE Din 习 


图 5.26 考题 信息 添加 页 面 的 运行 结果 
考题 信息 添加 页 面 涉及 到 的 HTML 表单 的 重要 元 素 如 表 5.8 所 示 。 


表 5.8 ”考题 信息 添加 页 面 涉及 到 的 HTML 表单 的 重要 元 素 


名 称 类 型 重要 属性 含义 
form2 form Imethod= post ”action= ktxx_tj ok.php” 表单 
<?php 
S$query=mssql query( select * from tb_ktlb”): 
while($myrow=mssql fetch array($query)){ 
kt lb select 本 考题 类 别 
<option value="*<?php echo $myrow[online ktlb]:?>”> 
<?php echo $myrow[online ktlb]:?> 
</option> 
php }7> 
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续 表 
名 称 类 型 重要 属性 含义 
<select> 
<option value=" 第 一 套 题 "> 第 一 套 题 </option> 
i a <option value=" 第 二 套 是 "> 第 一 夸 题 Joption> 所 属 套 是 
<option value=" 第 三 套 题 "> 第 三 套 题 </option> 
<option value=" 第 四 套 题 "> 第 四 套 题 </option> 
</select> 
<select> 
<option value="2"> 简 答 </option> 
让 i “option Vane >i depo 考题 类 型 
<option value="0"> 单 选 </option> 
<option value="1"> 多 选 </option> 
</select> 
kt fs text 考试 成 绩 
kt nr textarea 考题 内 容 
kt_daan textarea g 考题 答案 
kt_zqdaan textarea cols="60" rows= 考题 正确 答案 
Submit2 submit 计 “提交 考题 ”按钮 


添加 考试 题目 首先 要 选择 考题 的 类 别 ， 然 后 选择 所 属 套 题 ， 再 选择 考题 类 型 ， 其 实现 过 程 在 表 5.8 
所 示 的 表单 的 重要 属性 里 给 出 了 完整 的 代码 ， 这 里 不 再 歼 述 ， 最 后 根据 考题 类 别 将 考题 内 容 添加 完整 。 

提交 考题 表单 到 数据 处 理 页， 程序 处 理 页 面 首先 应 用 变量 获取 到 表单 数据 ， 然 后 应 用 insert...into 
语句 将 其 插入 到 考题 信息 表 tb_kt 中 。 如 果 考 题 信息 添加 成 功 ， 则 弹出 提示 信息 ， 并 重新 定位 到 考题 信 
息 添加 页 面 。 数 据 处 理 页 的 完整 代码 如 下 : 


例 程 14 ”代码 位 置 : 光盘 \TM\05\online_ks\admin\ktxx_tj_ok.php 


<?php 
include("../conn/conn.php"): // 连 接 数 据 库 文件 
if($Submit2 一 "提交 考题 "){ // 如 果 单 击 了 “提交 考题 ”按钮 ， 则 执行 下 面 语句 


$queryes=mssql query("insert into tb_kt 
(kt_1b,kt 1x,kt fs,kt nr.kt daan,kt zqdaan,kt small lb)values('$kt lb','$kt lx','$kt fs','$kt nr','$kt daan'.'$kt zqdaan','$kt small 


Im /向 考题 信息 表 中 添加 数据 
if($queryes){ /如 果 添 加 操作 成 功 ， 则 弹出 提示 ， 并 重 定向 网 页 
echo "<script>alert( 考 题 添加 成 功 ! ): window.location.href='index.php?htgl= 考 题 信息 添加 ':</script>"; 
中 
} 
?> 


5.9.4 ”查询 考题 信息 的 实现 过 程 


国 ”查看 考试 题目 列表 使 用 的 数据 表 : tb kt、tb ktlb 
管理 员 登 录 后 ， 单 击 “ 考 题 信息 管理 ” 超 链接 ， 进 入 到 查询 考题 信息 页 面 ， 选 择 考 题 类 别 后 ， 单 
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击 “考题 查找 ”按钮 ， 将 查询 出 该 类 别 下 的 所 有 考题 信息 。 同 时 提供 修改 考题 信息 和 删除 考题 信息 的 
功能 。 查 询 考题 信息 页 面 的 运行 结果 如 图 5.27 所 示 。 


会 在 本 考试 系 绞 后 全 管理 )》 考题 信息 管理 


| ER 可。 [二 


5.27 查询 考题 信息 页 面 的 运行 结果 


查询 考题 信息 页 面 涉及 到 的 HTML 表单 的 重要 元 素 如 表 5.9 所 示 。 


表 5.9 查询 考题 信息 页 面 涉及 到 的 HTML 表单 的 重要 元 素 


名 称 类 型 重要 属性 含义 
forml form method="post" action="index.php?htgl= 考 题 信 息 管理 " 查询 考题 表单 

<select name="kt lb" id="kt lb"> 

<?php 

$query=mssql query("select * from tb_ktlb"): 

while($Smyrow=mssql_fetch array($query)){ 
kt lb select ?> 考题 类 别 

<option value="<?php echo Smyrow[online ktlb]:?>"> 

<?php echo Smyrow[online ktlb]:?></option> 

<?php }°> 

</select> 
Submit submit value= "考题 查找 " “考题 查找 ”按钮 
form2 form method="post" action="ktxx_gl_ok.php” 考题 信息 表单 
kt lb text value="<?php echo Smyrow[kt 1b]:?>' 考题 类 别 显示 
kt lx text Value="< echo Smyrow[kt Ilx]:?>' 考题 类 型 
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续 表 
名 称 类 型 重要 属性 含 义 
kt fs text | value="<?php echo $myrow[kt fs]:?>" | 考试 成 绩 
kt id hidden | value="<?php echo Smyrow[kt id]?>" | 考题 类 别 id 
Submit? submit ”| value=" 修 改 " | “修改 ”按钮 
Submig submit | 。_value=" 删 除 " | “删除 ”按钮 
kt nr textarea <?php echo Smyrow[kt nr]:?> 考题 内 容 
kt daan textarea <?php echo Smyrow[kt daan]:?> 考题 答案 
kt zqdaan textarea <?php echo Smyrow[kt zqdaan]:?> 考题 正确 答案 


在 考试 题目 查询 页 面 中 ， 首 先 建立 用 于 查询 的 表单 forml， 该 表单 中 包含 “考题 类 别 ” 列 表 /菜单 
控件 以 及 “考题 查找 ”按钮 。 

当 管 理 员 选 择 考题 类 别 后 ， 单 击 “ 考 题 查找 ”按钮 ， 提 交 考 题 kt_lb 类 别 到 当前 页 ， 然 后 根据 获取 
到 的 考题 类 别 检索 考题 信息 表 tb_kt， 并 将 该 类 别 下 的 所 有 考题 信息 输出 到 浏览 器 ， 代 码 如 下 : 


例 程 15 ”代码 位 置 : 光盘 \TM\05\online_ks\admin\ktxx_gl.php 


ee -考题 信息 显示 - - - -一 一 

<table width="682" height="168" border="0" cellpadding="0" cellspacing="1" bgcolor="#5D554A"> 

<?php 

include("../conn/conn.php"): // 连 接 数 据 库 文件 
© $query=mssql query("select* from tb_ kt where kt lb='$kt lb"): /根据 选择 考题 类 别 检索 数据 信息 
© while($myrow=mssql fetch_ array(Squery)){ // 将 查询 结果 应 用 while 循 环 输出 

?> 

<form name="form?2" method="post" action="ktxx_gl] ok.php"> 

<tr> 
<!- -一 一 -一 一 考题 类 别 显示 一 一 一 一 一 一 一 一 一 一 一 -一 - -> 


<td width="112" height="27" align="center" bgcolor="#DDDDDD"> 考 题 类 别 </td> 
<td width="117" align="center" bgcolor="#DDDDDD"> 

<input name="kt_lb" type="text" value="<?php echo Smyrow[kt lb];?>" size="8"> 
</td> 
二 二 一 一 一 考 昨 类 型 星 示 二 -一 一- 三 = > 
<td width="180" align="center" bgcolor="#DDDDDD"> 考 题 类 型 

<input name="kt lx" type="text" value="<?php echo Smyrow[kt 1x];?>" size="10"> 
</td> 
EE 考试 成 绩 显示 - 一 > 
<td width="148" align="center" bgcolor="#DDDDDD"> 分 数 

<input name="kt_fs" type="text" value="<?php echo Smyrow[kt fs];?>" size="5"> 
</td> 
< 二 一 一 一 一 一 一 一 建立 考题 了 隐藏 域 ， 添 加 “修改 ”、“ 删 除 ”按钮 一 一 一 一 
<td width="99" rowspan="4" align="center" bgcolor="#FFFFFF"> 

<input type="hidden" name="kt id" value="<?php echo Smyrow[kt id]?>"> 

<input type="submit" name="Submit2" value=" 修 改 ">/<input type="submit" name="Submit3" value=" 删 除 "> 

</td> 


= 考题 内 容 显 示 一 


<td height="43" align="center" bgcolor="#DDDDDD"> 考 题 内 容 </td> 
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<td colspan="3" align="center" bgcolor="#FFFFFF"><textarea name="kt nr" cols="60" rows="5"><?php echo 
Smyrow[kt nr];?></textarea></td> 
</t> 
< 考题 答案 显示 一 
<t> 
<td height="46" align="center" bgcolor="#DDDDDD"> 考 题 答案 </td> 
<td colspan="3" align="center" bgcolor="#FFFFFF"><textarea name="kt daan" cols="60" rows="5"><?php echo 
Smyrow[kt_daan]:?></textarea></td> 
</u> 
<!— 考题 正确 答案 显示 一 
<t> 
<td height="33" align="center" bgcolor="#DDDDDD"> 考 题 正确 答案 </td> 
<td colspan="3" align="center" bgcolor="#FFFFFF"><textarea name="kt_zqdaan" cols="60" rows="5"><?php echo 
Smyrow[kt_zqdaan];?></textarea></td> 
</a> 


} //while 循 环 语句 结束 
E23 
‘</table> 


< 代码 贴 十 
@ mssql query0 函 数 : 向 SQL Server 服务 器 中 的 当前 活动 数据 库 发 送 一 条 SQL 查询 。 
@ mssql fetch_ array0 削 数 : 返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 语 法 如 下 : 
array mssql_fetch array( resource result [, int result_type] ) 
参数 result: 资源 类 型 的 参数 ， 要 传 入 的 是 由 mssql query0 函 数 返回 的 “数据 指针 ”。 
参数 Tesult_ type: 可 选项 , 整数 型 参数 ,要 传 入 的 是 MSSQL ASSOC、MSSQL NUM 和 MSSQL BOTH 3 种 由 PHP 
定义 好 的 常数 之 一 ， 默 认 值 是 MSSQL _BOTH. 
(1) 用 MSSQL ASSOC 只 得 到 关联 索引 。 
(2) 用 MSSQL NUM 只 得 到 数字 索引 。 
(3) 用 MSSQL BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 。 


查询 考题 信息 页 面 提供 了 修改 考题 信息 的 功能 ， 管 理 员 可 对 指定 的 考题 信息 进行 编辑 ， 单 击 “ 修 
改 ” 按 钮 后 ， 提 交 Submit2 表单 信息 到 数据 处 理 页 ktxx_gl_ok.php。 
例 程 16 ”代码 位 置 : 光盘 \TM\05\online_ks\admin\ktxx_g]_ok.php 


<?php 
include("../conn/conn.php"): // 连 接 数 据 库 文件 
if($Submit2—true){ // 如 果 单 击 了 “修改 ”按钮 ， 则 执行 下 面 语句 


用 update 语 名 修改 考题 信息 
$querys=mssql query("update tb_kt set 

kt lb='$kt lb',kt Ix='$kt Ix'kt fs='$kt fs'kt nr='$kt nr',kt daan='$kt daan'kt zqdaan='$kt zqdaan' where kt id='$kt id"): 
人 wweessswssrwsweessty 如 果 考 是 信息 修改 成 功 ， 弹 出 提示 ， 并 重 定向 到 首页 =wswwsswswswssswswswsw of 
这 3Squerys){ 
echo "<scripf>alert( 考 题 更 新 成 功 ! ): window.location href-'index.php?htgl=- 考 题 信息 管理 :</scripf>": 
by 
} 
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查询 考题 信息 页 面 提 供 了 删除 考题 信息 的 功能 ， 管 理 员 可 对 指定 的 考题 信息 进行 删除 ， 单 击 “ 删 
除 ” 按 钮 后 ， 提 交 Submit3 表单 信息 到 数据 处 理 页 ktxx_gl_ok.php。 


例 程 17 ”代码 位 置 ， 光盘 \TM\05\online ks\admin\ktxx_gl_okphp 


<?php 
include("../conn/conn .php"): // 连 接 数 据 库 文件 
if($Submit3—true){ // 如 果 单 击 了 “删除 ”按钮 ， 则 执行 下 面 语句 


/= 光 宙 宙 交 史册 灾 灾 宙 交 安安 克 克 宙 ww 用 delete 语 名 删除 指定 的 考题 信息 *** 和 和 */ 
$query=mssql query("delete from tb_kt where kt id='$kt id"); 

/* 于 ww 如 果 考 题 信息 删除 成 功 , 弹出 提示 ， 并 重 定向 到 首页 wwewwwewwwewwwwmwwemrwawww */ 
这 S$query){ 

echo "<script>alert(' 考 题 类 别 删除 成 功 ! ): window.location href-'index.php?htgl- 考 题 信息 管理 :</script>": 


?> 


5.10 开发 技巧 与 难点 分 析 


5.10.1 考生 登录 编号 的 获取 


考生 登录 编号 的 生成 主要 应 用 的 是 mt rand0 函 数 和 substr0 函 数 。 首 先 通过 mt rand() 函 数 来 获取 
一 个 100000 一 999999 之 间 的 随机 数 ， 然 后 应 用 substr0) 函 数 从 该 随机 数 中 获取 6 个 数字 ， 作 为 考生 
编号 。 

mt_rand0 函 数 的 功能 是 获取 随机 数 的 值 。 语 法 格式 如 下 : 

int mt rand([int min,int max]) 
[ 说明: 如 果 mt_rand0O 函 数 没有 提供 可 选 参数 min 和 max， 则 返回 0 到 RAND_MAX 之 间 的 伪 随 

机 数 。 
下 面 应 用 mt_rand0) 函 数 来 获取 100000 一 999999 之 间 的 随机 数 ， 代 码 如 下 : 
<?php echo mt rand(100000.999999): ?> 


运行 结果 为 : 321864 
5.10.2 ”通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 


通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 首 先 要 创建 一 个 XMLHttpRequest 对 象 实例 ， 确 保 其 能 够 
在 所 有 支持 XMLHttpRequest 的 浏览 器 中 运行 ,将 其 代码 保存 在 一 个 名 称 为 xmlHttpRequestjs 的 文件 中 ， 
然后 在 需要 应 用 Ajax 技术 的 页 面 中 ， 应 用 以 下 代码 包含 该 文件 。 


<script type="text/javascript" src="js/xmlHttpRequest.js"></script> 
xmlHttpRequestjs 文件 的 代码 如 下 : 
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例 程 18 代码 位 置 : 光盘 \TM\05\online_ ksijs\xmlHttpRequest.js 


var xmlHttp = false: // 创 建 一 个 布尔 型 变量 ， 用 来 检测 是 否 为 合法 的 正 实例 
ty{ // 检 测 是 否 使 用 的 是 正 

xmlHttp = new ActiveXObject("Msxml2.XMLHTTP"): /如 果 JavaScript 的 版 本 大 于 5 
} catch (e) { /如 果 不 是 ， 则 使 用 老 版 本 的 ActiveX 对 象 

ty{ // 如 果 使 用 的 是 正 浏览 器 


xmlHttp = new ActiveXObject("Microsoft. XMLHTTP"): 
} catch (e2) 他 


b 
/二 将 寺 站 于 机 让 站 站 下 本 相让 夺 和 让 本 如 果 使 用 的 是 非 IE 浏 览 器 ， 则 创建 一 个 该 对 象 的 JavaScript 实 例 兴 # 池 半 中字 相机 于 下 中字 下 囊 于 本 宙 下/ 


if (!IxmlHttp && typeof XMLHttpRequest != "undefined") { 
ty{ 
xmlHttp = new XMLHttpRequestO: 
}catch(e3){ xmlHttp = false:} 
} 


接 下 来 编写 两 个 自 定义 的 JavaScript 函数 ShowTime0O 和 sparetime()， 通 过 ShowTime0) 函 数 读 取 显 
示 时 间 文 件 ShowTime.php 中 的 数据 ; 通过 sparetime0 〇 函数 读 取 获取 剩余 时 间 文 件 sparetime.php 中 的 数 
据 。ShowTime0 和 sparetime0 函 数 的 代码 如 下 : 


例 程 19 代码 位 置 ; 光盘 \TMWOS\online_ ksWjs\xmlHttpRequestjs 
<script type="text/javascript"> 


timer = window.setInterval("ShowTime0".1000): // 每 隔 一 秒 钟 调用 一 次 ShowTime0 函 数 
/定义 ShowTime0 函 数 以 通过 xmlHttpRequest 对 象 读 取 ShowTime.php 文 件 中 的 数据 
function ShowTimeO{ 
XmlHttp.open("post"."showtime .php". true): // 以 post 方 法 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange = function|O{ 
这 xmlHttpreadyState — 4){ // 如 果 服 务 器 响应 发 出 的 请 求 ， 则 执行 以 下 操作 
tet = xmlHttp.responseText: // 获 取 返 回 的 响应 信息 


A 中 让 让 中 中 站 站 半 下 中 字 站 中 相让 中 站 本 站 本本 半 落 取 到 | 的 信息 赋予 指定 的 DIV 标 记 * 于 于 六 于 浊 六 六 素 水 站 半球 半 站 半 半 兴 站 家 */ 
document.getElementById("show_time").innerHTML = tet: 


} 
} 
xmlHttp.send(null); // 发 送 请 求 
} 
</script> 
‘<script type="text/javascript"> 
time = window.setInterval("sparetime()".1000): // 每 隔 一 秒 钟 调用 一 次 sparetime0 函 数 
六 yessstty 定 义 sparetimeO 函 数 以 通过 xmlHttpRequest 对 象 读 取 sparetime php 文 件 中 的 数据 **ssyyyysssst 4/ 
function sparetimegO{ 
xmlHttp.open("post","sparetime .php". true): // 以 post 方 法 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange = function|O{ 
if(xmlHttp.readyState 一 4){ // 如 果 服 务 器 响应 发 出 的 请 求 ， 则 执行 以 下 操作 


tet = xmlHttp.responseText: /获取 返回 的 响应 信息 
/二 本 本 半 可 二 半 训 二 站 可 本 叶 本 本寺 站 本 站 ## 半 获取 到 的 信息 赋予 指定 的 DIV 标记 地 检 二 李 本 李 李 本 可 本 吕 池 本 池村 可 池 于 于 浊 于 生 
document.getElementById("sparetime").innerHTML = tet:; 
if(tet—"00:00"){ // 判 断 当 剩 余 时 间 为 00:00 时 
forml.submitO: /提交 forml 表 单 中 的 数据 
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} 
本 
xmlHttp.send(null): // 发 送 请 求 
</script> 
在 ShowTime.php 文件 中 实现 当前 时 间 的 显示 。 实 现 的 原理 是 : 获取 一 个 在 session 变量 中 存储 的 
考试 开始 时 间 的 时 间 惟 ， 然 后 再 应 用 mktime0 函 数 获取 当前 时 间 的 时 间 截 ， 应 用 当前 时 间 截 减 去 考试 
开始 时 间 的 时 间 戳 ， 最 后 应 用 date0 函 数 输出 获取 的 新 时 间 戳 的 时 间 值 ， 代 码 如 下 ; 


例 程 20 ”代码 位 置 光盘 \TMWS\online ks\ShowTime php 


<?php session_start():; // 初 始 化 session 变 量 
header('Content-Type: text/xml"): 
Sdates=$_SESSION[dates]: /获取 session 变 量 中 存储 的 考试 开始 时 间 时 间 戳 
S$dates2=mktime(): /获取 系统 当前 时 间 的 时 间 惟 
S$dates3=$dates2-$dates: /应 用 当前 时 间 的 时 间 戳 减 去 考试 开始 时 间 的 时 间 戳 
echo date("i:s", $dates3); // 输 出 得 出 的 时 间 

?> 


在 sparetime.php 文件 中 获取 考试 的 剩余 时 间 。 实 现 的 原理 是 : 首先 设置 考试 时 间 为 20 分 钟 ， 在 考 
试 开始 时 间 的 基础 上 加 上 20 分 钟 , 然后 用 考试 时 间 减 去 系统 的 当前 时 间 , 获取 的 就 是 考试 的 剩余 时 间 ， 
当 剩余 时 间 为 00:00 时 表示 本 次 考试 结束 ， 将 考试 题 自动 提交 。sparetime.php 文件 的 代码 如 下 : 

例 程 21 代码 位 置 : 光盘 \TMVWSvonline_ ks\sparetime php 


<?php session_start(): /初始 化 session 变 量 
header(Content-Type: text/xml)): 
S$dates=$_ SESSION[dates]: /获取 session 变 量 中 存储 的 考试 开始 时 间 戳 
Sdatesl=$datest1*60; // 设 定 考试 时 间 
Sdates2=mktime(): /获取 系统 的 当前 时 间 戳 
S$dates3=$dates1-Sdates2: /计算 考试 的 剩余 时 间 
echo date("i:s".Sdates3): // 输 出 考试 的 剩余 时 间 
?> 
最 后 在 考试 操作 页 面 中 通过 DIV 标签 来 输出 考试 时 间 和 剩余 时 间 。 代 码 如 下 : 
<div id="sparetime"></div> <!-- 输 出 考试 剩余 时 间 --> 
<div id="show_time"></div> <!-- 输 出 考试 时 间 --> 


5.11 Ajax 无 刷新 技术 专题 


5.11.1 Ajax 概述 

Ajax 是 Asynchronous JavaScript and XML 的 缩写 ， 意 思 是 异步 的 JavaScript 与 XML。 

Ajax 并 不 是 一 种 新 技术 , 或 者 说 它 不 是 一 种 技术 , 实际 上 , 它 是 结合 了 Javascript XHTML 和 CSS、 
DOM、XML 和 XSTL、XMLHttpRequest 等 编程 技术 以 新 的 强大 方式 组 合 而 成 ， 可 以 让 开发 人 员 构建 
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基于 PHP 技术 的 Web 应 用 ， 并 打破 了 使 用 页 面 重 载 的 惯例 。Ajax 包含 : 
XHTML 和 CSS 技术 实现 标准 页 面 。 
Document Object Model 技术 实现 动态 显示 和 交互 。 
XML 和 XSLT 技术 实现 数据 的 交换 和 维护 。 
XMLHttpRequest 技术 实现 异步 数据 接收 。 
JavaScript 绑 定 和 处 理 所 有 数据 。 
Ajax 是 一 种 运用 浏览 器 的 技术 ， 它 可 以 在 浏览 器 和 服务 器 之 间 得 到 异步 通信 机 制 进行 数据 通信 ， 
从 而 允许 浏览 器 向 服务 器 获取 少量 信息 而 不 是 刷新 整个 页 面 。 


加 回回 回回 


5.11.2 Ajax 的 优点 


Ajax 是 使 用 客户 端 脚本 与 Web 服务 器 交换 数据 的 Web 应 用 开发 方法 。 这 样 ，Web 页 面 不 用 打 断 
交互 流程 进行 重新 加 载 ， 就 可 以 动态 地 更 新 。Ajax 优点 如 下 : 
(1) 减轻 服务 器 的 负担 。Ajax 的 原则 是 “ 按 需 取 数 据 ”， 可 以 最 大 程度 地 减少 元 余 请 求 ， 从 而 减 
轻 对 服务 器 造成 的 负担 。 
(2) 无 刷新 更 新 页 面 ， 减 少 用 户 心 理 和 实际 的 等 待 时 间 。“ 按 需 取 数据 ”的 模式 减少 了 数据 的 实 
际 读 取 量 。 如 果 说 重 载 的 方式 是 从 一 个 终点 回 到 原点 再 到 另 一 个 终点 的 话 〈 如 图 5.28 所 示 ) ， 那 么 Ajax 
就 是 以 一 个 终点 为 基点 到 达 另 一 个 终点 (如 图 5.29 所 示 ) 。 


7 oN oN 
@) @) (© 


图 5.28 ” 重 载 方式 5.29 Ajax 方式 


其 次 ， 即 使 要 读 取 较 大 的 数据 ， 也 不 会 出 现 白 屏 的 情况 。Ajax 使 用 XMLHTTP 对 象 发 送 请 求 并 得 
到 服务 器 响应 ， 在 不 重新 载 入 整个 页 面 的 情况 下 用 JavaScript 操作 DOM 最 终 更 新 页 面 ， 所 以 在 读 取 数 
据 的 过 程 中 ， 用 户 所 面 对 的 不 是 白 屏 ， 而 是 原来 的 页 面 状 态 ; 页 面 只 有 接收 到 全 部 数据 后 才 更 新 相应 
部 分 的 内 容 ， 而 这 种 更 新 也 是 瞬间 的 ， 用 户 几 乎 感觉 不 到 。 

(3) 带 来 更 好 的 用 户 体验 。 

(4) 把 部 分 服务 器 负担 的 工作 转交 给 客户 端 ， 利 用 客户 端 闲置 的 能 力 来 处 理 任 务 ， 从 而 减轻 服务 
器 和 带宽 的 负担 ， 节 约 空间 和 宽带 租用 成 本 。 

(5) 可 以 调用 外 部 数据 。 

(6) 是 一 种 基于 标准 化 并 被 广泛 支持 的 技术 ， 不 需要 下 载 插件 或 者 小 程序 。 

(7) 进一步 促进 Web 页 面 展 现形 式 与 数据 的 分 离 。 


5.11.3 Ajax 的 工作 原理 


传统 的 Web 模式 强制 用 户 进入 “提交 一 等 待 一 重新 显示 ”网 页 ， 用 户 的 动作 总 是 与 服务 器 进行 同 
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步 思考 ， 客 户 在 网 页 上 的 操作 转化 为 HTTP 请 求 传 回 服务 器 ， 而 服务 器 接受 请 求 以 及 相关 数据 、 解 析 
数据 并 将 其 发 送 给 相应 的 处 理 单元 后 ， 将 返回 的 数据 转 成 HTML 页 返还 给 客户 。 而 当 服 务 器 处 理 数据 
的 时 候 ， 用 户 只 能 等 待 ， 每 一 步 操 作 都 需要 等 待 服务 器 返回 新 的 网 页 。 由 于 每 次 应 用 的 交互 都 需要 向 
服务 器 发 送 请 求 ， 应 用 的 响应 时 间 就 依赖 于 服务 器 的 响应 时 间 ， 这 就 导致 了 用 户 页 面 的 响应 比 本 地 应 
用 慢 得 多 。 

运用 了 Ajax 技术 的 Web 应 用 模型 ， 它 的 工作 原理 相当 于 在 客户 端 和 服务 器 端 之 间 添 加 了 一 个 中 
间 层 ， 称 为 Ajax 引擎 (采用 JavaScript 编写 ， 通 常 在 一 个 隐藏 的 框架 中 ) ， 实 现 了 与 服务 器 进行 异步 
思考 的 通信 能 力 ， 从 而 使 用 户 从 请 求 /响应 的 循环 中 解脱 出 来 ， 向 服务 器 发 出 异步 请 求 ， 也 就 是 不 用 等 
待 服务 器 的 通信 。 所 以 用 户 不 用 再 打开 一 个 空白 窗口 ， 等 待 服务 器 完成 后 再 进行 响应 。Ajax 应 用 可 以 
仅 向 服务 器 发 送 并 取 回 必需 的 数据 ， 它 使 用 SOAP 或 其 他 一 些 基 于 XML 的 Web Service 接口 ， 并 在 客 
户 端 采用 JavaScript 处 理 来 自 服务 器 的 响应 。 因 为 在 服务 器 和 浏览 器 之 间 交 换 的 数据 大 量 减 少 ， 所 以 
Web 站 点 看 起 来 是 即时 响应 的 。 同 时 很 多 的 处 理工 作 可 以 在 发 出 请 求 的 客户 端 机 器 上 完成 ， 所 以 Web 
服务 器 的 处 理 时 间 也 减少 了 。 

引入 Ajax 的 Web 模型 与 传统 的 Web 模型 的 比较 如 图 5.30 所 示 。 


传统 Web 应 用 模型 Ajax Web 应 用 模型 
客户 端 


客户 端 


浏览 器 


JavaSeript 调 用 


HTML+CSS 数 据 


Ajax 引擎 


人 HTTP 请 求 
IHTML+CSS 数 据 


HTTP 服 务 器 


HTTP 或 XML 服 务 器 


应 用 服务 器 + 数据 库 服务 器 


应 用 服务 器 + 数据 库 服 务 器 


服务 器 端 


图 5.30 传统 Web 应 用 模型 与 Ajax Web 应 用 模型 的 比较 
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5.11.4 Ajax 的 工作 流程 


使 用 Ajax, 用 户 可 以 创建 接近 本 地 桌面 应 用 的 更 直接 、 更 可 用 、 更 丰富 、 更 动态 的 Web 用 户 界面 。 
Ajax 内 部 的 工作 流程 如 图 5.31 所 示 。 


de 


Intemet 用 户 Web 客户 端 


Web 浏 览 器 发 送 HTTP 请 求 给 服务 器 


Web 服 务 器 格 请 使 用 日 TML 和 JavaScript 员 应 仿生 
Web 服 务 器 


JavaScript 代 码 向 服务 器 发 送 不 可 见 调用 请 求 图 


服务 器 通过 发 送 
请 求 的 数据 进行 响应 

:更 新 各 户 端 显示 的 
内 容 


JavaScript 根 据 响 应 数据 更 新 Web 页 面 


5.31 Ajax 内 部 工作 流程 图 
5.11.5 ”Ajax 中 的 核心 技术 XMLHttpRequest 


Ajax 技术 中 最 核心 的 技术 就 是 XMLHttpRequest， 它 是 一 个 具有 应 用 程序 接口 的 JavaScript 对 象 ， 
能 够 使 用 超 文本 传输 协议 〈Http ) 连接 一 个 服务 器 ， 是 微软 公司 为 了 满足 开发 者 的 需要 ， 于 1999 年 在 
下 5.0 浏览 器 中 率先 推出 的 。 现 在 许多 浏览 器 都 对 其 提供 了 支持 ， 不 过 实现 方式 与 IE 有 所 不 同 。 

通过 XMLHttpRequest 对 象 ，Ajax 可 以 像 桌面 应 用 程序 一 样 只 同 服务 器 进行 数据 层面 的 交换 ， 而 
不 用 每 次 都 刷新 页 面 ， 也 不 用 每 次 都 将 数据 处 理 的 工作 交 给 服务 器 来 做 ， 这 样 既 减 轻 了 服务 器 负担 又 
加 快 了 响应 速度 、 缩 短 了 用 户 等 待 的 时 间 。 

在 使 用 XMLHttpRequest 对象 发 送 请 求 和 处 理 响应 之 前 , 首先 需要 初始 化 该 对 象 , 由 于 XMLHttpRequest 
不 是 一 个 W3C 标准 ， 所 以 对 于 不 同 的 浏览 器 ， 初 始 化 的 方法 也 是 不 同 的 。 

1，IE 浏览 器 

正 浏览 器 把 XMLHttpRequest 实例 化 为 一 个 ActiveX 对 象 。 具 体 方法 如 下 : 

var http_request = new ActiveXObject("Msxml?2.XMLHTTP"): 

或 者 


var http_request = new ActiveXObject("Microsoft XMLHTTP"): 


281 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


上 说 明 : 上 面 语法 中 的 Msxml2 XMLHTTP 和 Microsoft XMLHTTP 是 针对 IE 浏览 器 的 不 同 版 本 而 
进行 设置 的 ， 目 前 比较 常用 的 是 这 两 种 。 

2. Mozilla、Safari 等 其 他 浏览 器 

Mozilla、Safari 等 其 他 浏览 器 把 它 实例 化 为 一 个 本 地 JavaScript 对 象 。 具 体 代码 如 下 : 

Var http request= new XMLHttpRequest(): 

为 了 提高 程序 的 兼容 性 ， 可 以 创建 一 个 跨 浏览 器 的 XMLHttpRequest 对 和 象 。 创 建 一 个 跨 浏览 器 的 
XMLHttpRequest 对 象 其 实 很 简单 ， 只 需要 判断 一 下 不 同 浏览 器 的 实现 方式 ， 如 果 浏 览 器 提供 了 
XMLHttpRequest 类 ， 则 直接 创建 一 个 实例 ， 和 否则 使 用 正 的 ActiveX 控件 。 具 体 代码 如 下 : 

if (window.XMLHttpRequest) {/ Mozilla、Safari…… 


http request = new XMLHttpRequest(): 
} else if (window.ActiveXObject) {V/ 正 浏览 器 


ty{ 
http_request = new ActiveXObject("Msxml2.XMLHTTP"): 


} catch (e) { 


uy{ 
http_request = new ActiveXObject("Microsoft. XMLHTTP"); 


} catch (e) 人 


[a 说 明 : 由 于 JavaScript 具有 动态 类 型 特性 ， 而 且 XMLHttpRequest 对 象 在 不 同 浏览 器 上 的 实例 是 
兼容 的 ， 所 以 可 以 用 同样 的 方式 访问 XMLHttpRequest 实例 的 属性 的 方法 , 不 需要 考虑 创建 
该 实例 的 方法 是 什么 。 


5.12 本 章 总 结 


至 此 ， 一 个 完整 的 网 络 在 线 考试 系统 已 经 全 部 完成 。 在 程序 的 开发 过 程 中 ， 采 用 了 switch 框架 ， 
使 整个 系统 的 设计 思路 更 加 清晰 。 同 时 ， 为 了 使 程序 更 加 人 性 化 ， 系 统 中 应 用 了 Ajax 技术 实现 在 线 考 
试 时 自动 计时 等 功能 。 和 希望 读者 能 认真 学 习 ， 并 做 到 融会 贯通 。 


第 章 


物流 配送 信息 网 


(Apache+PHP+phpMyAdmin+MYySQL 5.0 实现 ) 
( 句 " 视频 讲解 : 1 小 时 27 分 钟 ) 


物流 信息 化 是 指 物流 企业 运用 现代 信息 技术 对 物流 过 程 中 产生 的 全 部 或 部 分 
信息 进行 采集、 分 类 、 传 遂 、 汇 总 、 坦 询 等 一 系列 处 理 活动 ， 以 实现 对 货物 流动 过 
程 的 控制 ， 从 而 降低 成 本 、 提 高 效益 的 管理 活动 。 

物流 企业 信息 化 的 目的 就 是 要 满足 企业 自身 管理 的 需要 和 不 同类 型 企业 在 物 
流 业 务 外 包 过 程 中 对 信息 交换 方 的 要 求 ， 也 就 是 通过 建设 物流 信息 系统 ， 提 高 信息 
流转 效率 ， 降 低 物 流 运作 成 本 。 

目前 我 国正 处 于 全 面 推进 信息 化 的 进程 之 中 , 所 以 物流 领域 的 信息 化 已 成 为 一 
个 必然 。 物 流 信息 化 将 成 为 现代 物流 的 灵 瑰 ， 将 是 现代 物流 发 展 的 必 经 之 路 。 

本 章 开 发 的 物流 配送 信息 网 就 是 为 了 增强 企业 的 市 场 竞争 力 ， 加 快 企业 的 信息 
化 进程 ， 使 企业 在 市 场 竞 争 中 立 于 不 败 之 地 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


物流 配送 系统 天 发 的 基本 过 程 

如 何 创 建 MySQL 数据 库 的 存储 过 程 

如 何 通过 MySQL 数据 库存 储 过 程 实现 用 户 登 录 
如 何 实现 数据 库 中 数据 的 模糊 查询 技术 

如 何 创 建 浮动 框架 

如 何 实现 订单 的 打印 技术 


豆 吾 吾 吾 吾 芋 
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6.1 开发 背景 


随 着 我 国信 息 化 进程 的 全 面 推 进 ， 各 领域 的 信息 化 进程 都 在 飞速 地 发 展 ， 同 样 也 推动 着 物流 领域 
的 信息 化 进程 飞快 地 向 前 发 展 。 由 于 信息 化 进程 地 全 面 推进 ， 对 现代 物流 提出 了 更 高 的 要 求 : 信息 化 、 
自动 化 、 网 络 化 、 智 能 化 和 和 柔性 化 等 。 物 流行 业 的 竞争 日 益 激烈 ， 客 户 需求 的 标准 也 越 来 越 高 ， 物 流 
企业 要 想 在 市 场 中 占有 一 席 之 地 ， 必 须要 建立 一 个 高 效 、 快 捷 、 方 便 的 物流 信息 系统 ， 为 客户 提供 一 
流 的 服务 ， 并 且 能 够 及 时 准确 地 掌握 客户 的 需求 ， 对 客户 的 需求 做 出 快速 的 反映 ， 在 最 短 的 时 间 内 以 
最 大 限度 挖掘 和 优化 物流 资源 来 满足 客户 的 需求 ， 从 而 建立 高 效 的 数字 化 物流 经 济 。 


6.2 系统 分 析 


6.2.1 需求 分 析 


随 着 现代 物流 信息 化 进程 的 加 快 ， 传 统 的 物流 管理 方式 已 经 不 再 适应 当前 物流 发 展 的 要 求 ， 取 而 
代 之 的 将 是 以 计算 机 为 基础 的 网 络 化 物流 管理 方式 。 某 物流 配送 公司 为 适应 物流 信息 化 进程 的 发 展 ， 
急需 开发 一 个 物流 配送 系统 ， 通 过 网 络 来 实现 对 物流 操作 流程 进行 管理 ， 不 但 为 企业 的 运营 过 程 节省 
大 量 的 人 力 、 物 力 、 财 力 和 时 间 ， 提 高 物流 系统 运行 的 效率 ， 而 且 为 企业 在 客户 中 树立 一 个 全 新 的 形 
象 ， 为 企业 的 发 展 黄 定 一 个 良好 的 基础 。 现 根据 对 该 物流 配送 公司 的 实际 调查 ， 以 及 公司 的 具体 要 求 ， 
制定 出 物流 配送 信息 网 的 规划 方案 。 具 体内 容 如 下 
网 站 页 面 设计 要 求 美观 大 方 ， 能 够 展示 企业 形象 。 
为 企业 在 客户 中 树立 一 个 全 新 的 形象 。 
网 站 的 操作 流程 简单 、 方 便 ， 能 够 提高 工作 效率 。 
提供 物流 配送 的 全 程 跟踪 。 
提供 配送 信息 的 及 时 查询 。 
实现 对 配送 车 辆 的 管理 。 
实现 对 客户 信息 的 管理 。 
实现 发 货 单 打印 的 功能 。 


办 办 办 欠 办 办 多 轨 


6.2.2 ”可行 性 分 析 


物流 配送 信息 网 的 开发 不 但 能 使 物流 企业 走 上 科学 化 、 网 络 化 管理 的 道路 ， 而 且 能 够 为 企业 带 来 
巨大 的 经 济 效益 和 实现 技术 上 的 飞速 发 展 。 


1. 经 济 性 
科学 的 管理 方法 ， 便 捷 的 操作 环境 ， 系 统 的 经 营 模式 ， 将 为 企业 带 来 更 多 的 客户 资源 ， 帮 助 企业 
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树立 品牌 形象 ， 提 高 企业 的 经 济 效益 。 
2. 技术 性 
网 络 化 的 物流 管理 方式 ， 在 操作 过 程 中 能 够 快捷 地 查找 出 车 源 信息 、 客 户 订单 以 及 客户 信息 ; 能 
够 对 货物 进行 全 程 跟 踪 ， 了 解 货物 的 托运 情况 ， 从 而 使 企业 能 够 根据 实际 情况 ， 做 好 运营 过 程 中 的 各 
项 准备 工作 ， 作 出 及 时 准确 的 调整 ， 能 够 保证 托运 人 以 及 收 货 人 对 货物 进行 及 时 的 处 理 。 


6.3 系统 设计 


6.3.1 系统 目标 


结合 目前 网 络 上 物流 配送 系统 的 设计 方案 ， 对 客户 做 的 调查 结果 以 及 企业 的 实际 需求 ， 本 项 目 在 
设计 时 应 该 满足 以 下 目标 : 
界面 设计 美观 大 方 、 操 作 简单 。 
功能 完善 、 结 构 清 晰 。 
能 够 快速 查询 车 源 信息 。 
能 够 准确 填写 订单 。 
能 够 实现 订单 查询 、 打 印 。 
能 够 实现 对 回 单 处 理 。 
能 够 对 车 源 信息 进行 添加 、 修 改 和 删除 。 
能 够 对 客户 信息 进行 管理 。 
能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
良好 的 数据 库 系 统 支持 。 
系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 


6.3.2 系统 功能 结构 


办 办 办 办 办 多 多 办 办 多 罗 


结合 需求 分 析 和 系统 目标 中 的 内 容 ， 物 流 配送 信息 网 的 功能 结构 已 经 设计 完成 。 为 了 使 读者 能 够 
更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 物流 配送 信息 网 的 功能 模块 结构 图 和 工作 流程 图 。 

物流 配送 信息 网 的 功能 模块 结构 图 如 图 6.1 所 示 。 

物流 配送 信息 网 的 工作 流程 图 如 图 6.2 所 示 。 


6.3.3 ”系统 预览 
物流 配送 信息 网 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 


物流 配送 信息 网 主页 如 图 6.3 所 示 ， 该 页 面 用 于 展示 本 系统 的 车 源 信息 查询 模块 。 登 录 页 面 如 图 6.4 
所 示 。 
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回执 发 货 单 确认 模块 的 页 面 效 果 如 图 6.5 所 示 , 该 页 面 主要 用 于 实现 对 回执 发 货 单 的 处 理 。 发 货 单 
查询 模块 的 页 面 效 果 如 图 6.6 所 示 , 该 页 面 主要 用 于 实现 对 指定 发 货 单 的 查询 , 并 且 输 出 发 货 单 的 详细 
内 容 。 

车 源 信息 管理 模块 的 页 面 效 果 如 图 6.7 所 示 , 该 页 面 实现 对 车 源 信息 的 添加 、 修 改 和 删除 操作 。 修 
改 管理 员 密 码 模块 的 页 面 效 果 如 图 6.8 所 示 ， 该 页 面 实 现 对 管理 员 密 码 的 修改 。 


一 | 添加 车 源 信息 

查询 车 源 信息 
| 修改 车 源 信息 

使 用 该 车 
-| 删除 车 源 信息 


须 全 长 站 涉 


货物 到 达 


6.2 ”物流 配送 信息 网 的 工作 流程 图 
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物流 配送 从 息 网 


用 户 名 :ee 密码 ; a5 [5] 


6.4 ”登录 页 面 (光盘 \TM\06\Wwlgl\index.php) 


6.7 车 源 信息 管理 (光盘 \TM\06\wlgl\car.php) 6.8 修改 管理 员 密码 (光盘 \TM\06\ wlgl\update_pass.php) 


o 
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6.3.4 ”开发 环境 


在 开发 物流 配送 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
。 服务器 端 
操作 系统 : Windows XP Server/Linux (推荐 ) 。 
服务 器 : Apache 6.0。 
PHP 软件 : PHP 5.0。 
数据 库 : MySQL 5.0。 
MySQL 图 形 化 管理 软件 : PhpMyAdmin-6.9.0。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 ， 最 佳 效果 1024X768 像素 。 
客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 


一 


办 办 办 办 多 办 多 


区 尽 守 


6.3.5 ”文件 夹 组 织 结构 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 
的 分 类 ， 将 其 分 别 放置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架 构 的 规划 ， 可 以 确保 网 站 文件 目录 明 
确 、 条 理 清晰 ， 同 样 也 便于 网 站 后 期 的 更 新 和 维护 。 物 流 配 送信 息 网 的 文件 夹 组 织 架构 如 图 6.9 所 示 。 


Ee “ld 
comm 


5 ss 
image 一 一 存 守 图 片 六 件 的 文件 
并 一 一 一 一 一 一 一 一 一 一 存储 肚 本 文件 的 文件 夹 


图 6.9 文件 夹 组 织 结构 
6.4 数据 库 设计 


物流 配送 信息 网 必须 拥有 数据 库 的 支持 ， 所 有 物流 配送 的 数据 都 应 该 存储 到 数据 库 中 ， 便 于 管理 
员 查 找 车 辆 、 订 单 和 客户 的 信息 。 如 果 没 有 数据 库 的 支持 ， 那 么 物流 配送 信息 网 将 没有 任何 意义 。 本 
节 将 对 物流 配送 信息 网 的 数据 库 设 计 进 行 详细 介绍 。 
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6.4.1 数据 库 分 析 


物流 配送 信息 网 是 一 个 中 小 型 的 企业 管理 系统 ， 考 虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 操作 的 灵 
活性 等 问题 ， 使 用 MySQL 数据库 是 最 佳 的 选择 。MySQL 数据 库 是 完全 免费 的 ， 使 用 它 不 需要 任何 费 
用 ， 可 以 直接 从 网 上 免费 下 载 ， 使 用 MySQL 数据 库 的 操作 也 非常 方便 ， 不 但 可 以 在 命令 模式 下 进行 ， 
而 且 配 备 目前 比较 流行 的 图 形 化 管理 工具 phpMyAdmin, 能 够 轻松 地 实现 对 MySQL 数据 库 的 管理 和 
操作 。 


6.4.2 ”数据 库 概念 设计 


根据 上 述 各 节 对 物流 配送 信息 网 做 的 需求 分 析 和 系统 设计 , 整理 出 物流 配送 信息 网 的 实体 关系 E-R 
图 。 其 中 包括 管理 员 信息 实体 、 车 源 信息 实体 、 车 辆 日 志 信息 实体 、 客 户 信息 实体 和 发 货 单 信息 实体 。 


1， 管理 员 信息 实体 


管理 员 信 息 实 体 用 于 存储 管理 员 的 登录 名 称 和 密码 信息 ， 包 括 用 户 名 和 密码 两 项 内 容 。 管 理 员 信 
息 实 体 的 E-R 图 如 图 6.10 所 示 。 


用 户 名 管理 员 信息 


图 6.10 管理 员 信息 实体 的 E-R 图 
2. 车 源 信息 实体 


车 源 信息 实体 用 于 存储 企业 拥有 的 车 辆 信息 ， 包 括 车 主 姓名 、 车 主 身份 证 号 码 、 和 车牌 号码、 车 主 
联系 电话 、 车 主 的 家 庭 地 址 、 车 辆 行驶 的 路 线 和 车 辆 描述 。 车 源 信息 实体 的 E-R 图 如 图 6.11 所 示 。 


车 主 家 庭 地址 


6.11 车 源 信息 实体 的 E-R 图 
3. 车 辆 日 志 信息 实体 
车 辆 日 志 信息 实体 记录 车 辆 当前 是 否 被 占用 ， 以 及 使 用 的 时 间 、 执 行 的 任务 ， 包 括 车 牌号 码 、 车 
辆 日 志 信息 〈 详 细 的 车 辆 使 用 描述 ) 、 日 志 时 间 和 发 货 单 ID 〈 即 执行 的 任务 ) 。 车 辆 日 志 信息 实体 的 
E-R 图 如 图 6.12 所 示 。 
4. 客户 信息 实体 
客户 信息 实体 用 于 存储 客户 的 信息 ， 包 括 客 户 的 姓名 、 客 户 电 话 和 客户 的 联系 地 址 。 客 户 信息 实 
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体 的 E-R 图 如 图 6.13 所 示 。 


车 牌号 码 
车 辆 日 志 信息 


图 6.12 ”车辆 日 志 信息 实体 的 E-R 


客户 姓名 
客户 电话 客户 信息 | 一 << 客户 联系 地 址 > 


图 6.13 客户 信息 实体 E-R 图 


5. 发 货 单 信息 实体 


发 货 单 信息 实体 用 于 存储 客户 填写 的 发 货 单 信息 ， 包 括 车 牌号 码 、 车 主 电话 、 货 物 描述 、 发 货 人 、 


发 货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 发 货 单 信息 实体 的 
E-R 图 如 图 6.14 所 示 。 


图 6.14 发 货 单 信息 实体 的 E-R 图 


6.4.3 创建 数据 库 及 数据 表 


在 物流 配送 信息 网 中 应 用 的 是 db_wlgl 数据 库 ， 其 中 涉及 5 个 数据 表 ， 如 图 6.15 所 示 。 
加 服务 器 : localhost ， 力 数据 库 : db_wlgl 


表 类 型 整理 说 明 
也 _admin MySAM ”gb2312_chinese_ci 管理 员 信息 表 
th_car MyISAM gb2312_chinese_ci 车 源 信息 表 
tb_car_log 。 MyISAM ”gb2312_chinese_ci 车辆 日 志 信 息 表 
了 共 _customer MyISAM gb2312_chinese_ci 客户 信息 表 
也 _shopping ”MyISAM 。 gb2312_chinese_ci 发 货 单 信息 表 


图 6.15 物流 配送 信息 网 的 数据 库 

下 面 介绍 物流 配送 信息 网 中 使 用 的 数据 表 ， 数 据 表 的 设计 结构 如 图 6.16 一 图 6.20 所 示 。 
1. tb_admin (管理 员 信息 表 ) 

管理 员 信息 表 主 要 用 于 存储 管理 员 的 登录 名 和 密码 。 该 数据 表 的 结构 如 图 6.16 所 示 。 


第 6 章 物流 配送 信息 网 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


胃 服务 器 : localhost 名 数据 库 : db_wlgl * 国 表 :tb_admin 


字段 类 型 整理 屋 性 Null 笃 认 秆 外 说 明 
过 int(10) auto_increment 管理 员 信 息 表 的 主键 
admin user varchar(50) gb2312_chinese_ci 否 登录 名 称 
admin_pass varchar(50) gb2312_chinese_ci 否 号 录 密码 


图 6.16 管理 员 信息 表 结 构 


2. tb_car (车 源 信 息 表 ) 


车 源 信息 表 主 要 用 于 存储 配送 公司 拥有 的 车 辆 信息 ， 主 要 包括 车 主 姓名 、 车 主 身份 证 号 码 、 车 牌 
号 码 、 车 主 联系 电话 、 车 主 的 家 庭 地 址 、 车 辆 行驶 的 路 线 和 车 辆 描述 。 该 数据 表 的 结构 如 图 6.17 所 示 。 


图 服务 器 : localhost ， 轧 数据 库 : db_wlgl ， 国 表 :tb_car 


字段 类 型 整理 屋 性 Null 默认 和 格外 说 明 
ud int(10) 否 auto_increment 车 源 信息 表 的 主键 
username varchar(50) gb2312_chinese_ci 否 车 主 的 姓名 
user_number varchar(50) gb2312_chinese_ci 理 车 主 的 身份 证 号 码 
car_number varchar(50) gb2312_chinese_ci 理 车 牌号 码 
tel varchar(50) gb2312_chinese_ci 否 车 主 电话 
address varchar(80) gb2312_chinese_ci 香 车 主 联系 地 址 
car_road mediumtext gb2312_chinese_ci 否 车 辆 行驶 路 线 
carcomtent mediumtedt gb2312_chinese_ci 理 车 辅 描述 


图 6.17 车 源 信息 表 结构 


3. tb_car_log (车辆 日 志 信 息 表 ) 


车 辆 日 志 信息 表 主要 用 于 存储 车 辆 的 使 用 情况 信息 ， 主 要 包括 车 牌号 码 、 车 辆 使 用 日 志 、 日 志 创 
建 时 间 和 车 辆 执行 的 任务 。 该 数据 表 的 结构 如 图 6.18 所 示 。 


图 服务 器 : localhost ， 属 数据库: db_wlgl ， 国 表 :tb_car_log 
字 及 类 型 整理 属性 Nul 默认 额外 说 明 


log id int(10) 否 auto_increment 车 辆 日 志 表 的 ID 
car_number varchar(50) gb2312_chinese_cl 理 车 牌号 码 
car_log mediumtext gb2312_chinese_el 否 车 畏 日 志 信息 
log_date datetime 否 日 志 包 娃 时 间 
fahuo_id varchar(50) gb2312_chinese_ci 理 发 货 单 的 ID 


6.18 ”车辆 日 志 信息 表 结构 


4. tb_customer (客户 信息 表 ) 
客户 信息 表 主 要 用 于 存储 客户 的 信息 ， 包 括 姓名 、 电 话 以 及 联系 地 址 。 该 数据 表 的 结构 如 图 6.19 


所 示 。 


踢 服务 器 : localhost 》 章 数据 库 : db_wlgl ， 国 表 :tb_customer 

字段 类 型 整理 屋 性 Null 默认 医 外 说 明 
customer id int(10) 否 auto_increment ”客户 信息 表 的 ID 
customer_user varchar(50) gb2312_chinese_ci 否 客户 姓名 
customer tel varchar(50) gb2312_chinese_ci 否 客户 电话 
customer_address varchar(80) 9b2312_chinese_ci 理 联系 地 址 


图 6.19 客户 信息 表 结构 
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5. tb_shopping (发 货 单 信息 表 ) 
发 货 单 信息 表 主 要 用 于 存储 客户 填写 的 发 货 单 中 的 信息 ， 主 要 包括 车 牌号 码 、 车 主 电 话 、 货 物 描 
述 、 发 货 人 、 发 货 时间 、 发 货 人 电话 、 发 货 地址、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 该 数 


据 表 的 结构 如 图 6.20 所 示 。 


加 服务 器 : localhost ， 电 数据 库 : db_wlgl 》 国 表 :tb_shopping 
字 及 类 型 整理 尾 性 Null 革 认 额外 


过 int(10) 
car_number varchar(50) 
fahuo_content mediumtex 


fahuo_id varchar(50) 
fahuo_user varchar(50) 
Tahuo_time datetime 

fahuo_ys varchar(20) 
Tahuo_fk varchar(20) 
car_tel varchar(50) 


Shouhuo_user varchar(50) 
shouhuo_address 。 mediumitext 
fahuo_address mediumtext 
fahuo_tel Varchar(50) 
shouhuo_tel varchar(50) 


gb2312_chinese_ci 
gb2312_chinese_ci 
9b2312_chinese_ci 
gb2312_chinese_ci 


9b2312_chinese_cl 
gb2312_chinese_ci 
gb2312_chinese_cl 
gb2312_chinese_ci 
gb2312_chinese_ci 
gb2312_chinese_cl 
gb2312_chinese_ci 
gb2312_chinese_ci 


天 呈 玉 哺 呈 呈 天 和 吓 别 荫 喇 葬 


说 明 
auto_increment ”发 货 单 信息 表 的 主妇 ID 


6.20 发 货 单 信息 表 结构 


6.5 网 站 首页 设计 


6.5.1 网 站 首页 概述 


网 站 首页 是 整个 网 站 的 门面 ， 是 浏览 者 看 到 的 第 一 视觉 界面 ， 所 以 在 设计 网 站 的 首页 时 应 该 将 网 
站 中 主要 的 内 容 尽 量 展示 给 浏览 者 ， 让 浏览 者 能 够 更 快 地 了 解 网 站 的 内 容 。 物 流 配 送信 息 网 的 首页 主 


要 包括 如 下 功能 : 


办 办 办 办 办 办 罗 


车 源 信息 查询 : 主要 实现 车 辆 信息 查询 ， 及 时 为 客户 选择 货物 配送 的 方案 。 
发 货 单 : 主要 用 于 客户 发 货 单 信息 填写 。 

回执 发 货 单 确认 : 主要 用 于 对 货物 配送 回执 单 的 确认 。 

发 货 单 查询 : 主要 用 于 对 发 货 单 进行 查询 。 

客户 信息 管理 : 主要 用 于 对 客户 的 信息 进行 管理 。 

车 源 信息 管理 : 主要 用 于 对 企业 拥有 的 车 辆 信息 进行 管理 。 

修改 密码 : 主要 用 于 对 管理 员 登 录 的 密码 进行 修改 。 


下 面 看 一 下 本 案例 中 提供 的 网 站 首页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\06\wlgl\index.php， 


如 图 6.21 所 示 。 
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网 站 Flash 头 文件 
修改 曾 理 员 密码 
FE 一 
请 
后 一 


功能 模块 内 容 展示 


图 6.21 物流 配送 信息 网 首页 


6.5.2 ”网 站 首页 技术 分 析 


物流 配送 信息 网 是 为 物流 配送 公司 设计 的 一 个 物流 管理 系统 ， 其 页 面 的 设计 突出 简洁 、 方 便 ， 功 
能 的 实现 以 便于 操作 和 维护 为 根本 。 网 站 首页 的 页 面 设计 如 图 6.21 所 示 ， 由 一 个 简单 的 框架 组 成 ， 其 
结构 由 两 部 分 组 成 : 一 部 分 用 于 输出 网 站 中 包括 的 模块 ， 另 一 部 分 用 于 输出 对 应 模块 中 的 内 容 。 首 页 
框架 结构 如 图 6.22 所 示 。 


输出 功能 模块 的 内 容 


6.22 首页 框架 结构 


这 里 使 用 的 框架 结构 不 是 frame 框架 ， 而 是 应 用 switch 语句 ， 加 上 表格 的 嵌 套 实现 的 框架 效果 。 
有 关 switch 语句 的 详细 应 用 可 以 参考 2.12.2 节 相关 内 容 。 
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6.5.3 ”网 站 首页 的 实现 过 程 


物流 配送 信息 网 的 首页 设计 非常 简单 ， 主 要 应 用 一 个 简单 的 框架 功能 ， 并 且 以 超 链接 作为 辅助 ， 
最 终 实现 在 不 同 功能 模块 之 间 的 跳 转 。 

首先 连接 数据 库 , 判断 登录 用 户 的 身份 , 如 果 是 管理 员 则 可 以 进入 网 站 进行 操作 ,否则 将 提示 “请 
您 正确 登录 ! ”， 并 且 跳 转 到 网 站 的 登录 页 面 。 关 键 代 码 如 下 : 

例 程 01 代码 位 置 : 光盘 \TM\06\wlgl\indexs.php 


<2php session start(); // 初 始 化 session 变 量 
include("conn/conn.php"): // 连 接 数 据 库 

if($_SESSION[admin user] 一 true){ // 判 断 登 录用 户 的 身份 

?> 

<!--> 省 略 了 部 分 代码 <!--> 

<?php 

}else{ // 没 有 以 管理 员 登 录 ， 则 跳 转 到 登录 页 


echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 
} 


?> 


然后 设计 首页 中 使 用 的 模块 超 链 接 ， 并 且 为 每 个 超 链接 定义 变量 标识 ， 当 单 击 不 同 功能 模块 超 链 
接 时 提交 不 同 的 变量 标识 ， 作 为 下 一 步 获取 模块 中 详细 内 容 的 依据 ; 其 中 还 使 用 ononMouseOver 和 
onMouseOut 事件 ， 控 制 鼠 标 移动 到 和 离开 单元 格 时 显示 不 同 的 颜色 。 部 分 代码 如 下 : 


到 应 用 onMouseOver 和 onMouseOut 事 件 控制 鼠标 移 过 和 离开 时 的 时 间 ------------------------ -- > 
<tr onMouseOver="this.bgColor=#9FCB3A"onMouseOut="this.bgColor=#F5F5F5" > 
<td height="28" align="center">&nbsp:</td> 
二 --- 设 置 车 源 信息 查询 模块 的 超 链接 ， 变 量 标识 为 "mbs- 车 源 信息 查询 "------------------ = 
<td align="left"><a href="indexs.php?lmbs= 车 源 信息 查询 "> 车 源 信息 查询 </a></td> 
</t> 
<tr onMouseOver="this.bgColor=-'#9FCB3A'"onMouseOut="this.bgColor=-#FSFSF5" > 
<td height="28" align="center">&nbsp:</td> 
< -设置 发 货 单 模块 的 超 链 接 ， 变 量 标识 为 "Imbs= 发 货 单 "---------------------------- -- > 
<td align="left"><a hre 仁 "indexs.php?lmbs= 发 货 单 "> 发 货 单 </a></td> 
</t> 


最 后 应 用 switch 语句 ， 根 据 变量 标识 $lmbs 提交 的 值 进行 判断 ， 使 用 include 包含 语句 调用 不 同 功 
能 模块 的 脚本 文件 。 关 键 代码 如 下 : 


<2php 
switch($lmbs){ // 获 取 变 量 $Slmbs 的 值 
case "车 源 信息 查询 ": // 浏 断 当 变量 Slmbs 的 值 是 “车 源 信息 查询 ”时 输出 下 面 的 内 容 
Ne select.php"): /调用 指定 的 包含 文件 
1/ 跳出 循环 
/让 省 部 分 代码 
case™ // 判 断 当 变量 的 值 为 空 时 
a car_select.php"): // 当 变量 的 值 为 空 时 则 调用 该 文件 
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break: // 跳 出 循环 


6.6 车 源 信息 查询 模块 设计 


6.6.1 车 源 信 息 查 询 模块 概述 


车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 货物 配送 的 地 点 , 从 数据 库 中 查询 有 关 该 路 线 的 车 辆 ， 
为 客户 提供 一 个 合理 配送 的 路 线 ， 最 后 确定 发 货 订 单 。 操 作 流 程 如 图 6.23 所 示 。 


图 6.23 车 源 信息 查询 的 操作 流程 


6.6.2 车 源 信息 查询 模块 技术 分 析 


车 源 信息 查询 模块 的 主要 功能 就 是 查询 车 辆 的 使 用 信息 ， 为 客户 提供 最 合适 的 货物 配送 路 线 。 其 
中 应 用 的 关键 技术 自然 就 是 查询 方法 ， 为 了 给 客户 提供 最 合适 、 最 满意 、 最 快捷 的 配送 服务 ， 这 里 使 
用 的 是 模糊 查询 技术 。 

通过 模糊 查询 技术 ， 只 要 客户 提出 配送 货物 的 出 发 地 点 和 到 达 地 点 ， 管 理 员 就 可 以 从 数据 库 中 提 
取出 所 有 与 该 路 线 相关 的 车 辆 信息 ， 包 括 车 辆 的 承载 能 力 、 车 辆 是 否 被 占用 和 车 辆 的 使 用 情况 ， 都 一 
目 了 然 ， 客 户 可 以 根据 实际 的 情况 〈 时 间 、 货 物 数 量 、 路 线 等 ) 选择 车 辆 ， 确 认 后 填写 发 货 单 。 

模糊 查询 技术 的 实现 是 在 car_select.php 文件 中 完成 的 ， 首 先 要 与 数据 库 建立 连接 ， 然 后 创建 一 个 
form 表单 ,设置 两 个 文本 框 selectl 和 select2 用 于 提交 出 发 地 点 和 到 达 地 点 ,将 表单 的 值 提交 到 当前 页 ; 
最 后 编写 PHP 脚本 语句 ， 以 表单 中 获取 的 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 
符合 条 件 的 信息 ， 并 且 将 查询 出 的 信息 显示 到 页 面 中 。 程 序 的 关键 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 \TMVWO6vwlglvcar_ selectphp 


<?php 
Session start(): include("conn/conn .php"): // 连 接 数据 库 
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?> 
<!-- 一 -一 一 一 一 创建 form 表 单 ， 提 交 出 发 地 点 和 到 达 地 点 ----- 
<form id="form1" name="form1" method="post" action="indexs.php?lmbs= 车 源 信息 查询 " onSubmit="javascript: return 


check select_car0:"> 
<td height="30" colspan="4" align="right" bgcolor="#FFFFFF"> 查 询 
© <input name="select1" type="text" id="select1" size="10"> 至 
© <input name="select2" type="text" id="select2" size="10"></td> 
<td colspan="2" bgcolor="#FFFFFF"><input type="submit" name="Submit" value=" 提 交 " /></td> 
</form> 
Ne 获取 表单 提交 的 值 ， 根 据 该 条 件 执行 ke 模糊 查询 ， 从 数据 库 中 读 取 数 据 ---…---------------- -- > 


日 ifSselectl 一 true || $select2—true){ /判断 表单 提交 的 值 是 否 为 真 
/+ +## 汪 二 和 二 和 和 下 ##### [人 表单 提交 的 值 为 条 件 ， 从 数据 表 中 查询 数据 #+#####+##y#tssss 4 

@ Squery="select * from tb_car where car road like '%Sselect1%" and car_ road like '%Sselect2%"; 
Sresult=mysql_query($query): // 执 行 查询 语句 
if(mysql_num rows(S$result)>0){ // 判 断 是 否 存 在 符合 条 件 的 数据 
while($Smyrow=mysql_fetch_array(S$result)){ // 应 用 while 循 环 语句 ， 输 出 符合 条 件 的 数据 


?> 


<td height="26" align="center" bgcolor="#FFFFFF"><?php echo $myrow[car_ number];?></td> 

<td bgcolor="#EFFFFF"><?php echo $myrow[car road]:?></td> 

<td colspan="2" bgcolor="#FFFFFF"><?php echo $myrow[car_content]:?></td> 

<td bgcolor="#FFFFFF"> 

<?php 
/* #4 根据 车 牌号 码 ， 从 车 辆 日 志 信 息 表 中 读 取 对 应 车 辆 的 使 用 信息 ********* 4/ 
$querys=mysql_query("select * from tb_car log Where car number='$myrow[car number]"): 
Smyrows=mysql_fetch_array($querys); 


echo Smyrows[car log]: // 输 出 车 辆 的 使 用 日 志 信息 

?> 

</td> 

<td align="center" bgcolor="#FFFFFF"> 

<?php 

if($myrows[car_log]=—nul){ /判断 车 辆 日 志 是 否 为 室 ， 如 果 为 室 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?lmbs= 发 货 单 &ljid=$Smyrow[id]> 使 用 该 车 </a>"; 

jelse{ /如 果 车 辆 日 志 不 为 室 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?lmbs= 发 货 单 &ljid=$myrow[id]> 预 定 该 车 </a>"; 

1 

?> 

</td> 

</tr> 


<?php }}else{ echo "您 查找 的 路 线 不 存在 !": }}?> 


< 人 代码 贴 二 
@ selectl: 指定 出 发 地 点 。 
@ select2: 指定 到 达 地 点 。 
四 证 语句 : 判断 从 表单 中 获取 的 变量 是 否 为 真 。 
@ like: 执行 模糊 查询 ， 从 数据 库 中 读 取 符合 条 件 的 数据 。 
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6.6.3 车 源 信 息 查 询 模块 的 实现 过 程 


国 车 源 信息 查询 模块 使 用 的 数据 表 : tb_car、tb_car log 
车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 信息 ， 从 数据 库 中 读 取 有 关 某 个 路 线 的 车 辆 使 用 的 情 
形 ， 及 时 为 客户 确定 一 个 合理 的 物流 配送 方案 。 该 模块 的 运行 结果 如 图 6.24 所 示 。 
二 ww] :3 | 


PEI SONG XIN XI 


EY 二 sl | 

ee ae) Ea ET 全 有 昌 志 。。 | 是 全 
刁 扫 发 代 草 确 认 总 | 长春 - 议 间 -大连 关于 . 下 太后 区 1 12 2 | me 
发 让 间 查 光 a rr 关机 寺 下 三 3 6 后 直径 (12 末 2 人 EE 
te | 医生 以 用 -大 加 | 天 车, 直 色 3 86 六 ,后 直 色 4_12 洒 ,并 24 汶 ， Es 

大 车. 下 芭 3. 6 后 下 本 412 末 长 2 

和 和 A-12545 长春- 沈阳 -大 连 -这 大。 | 雪 半 二 EE 
各 BR 尖 信息 网 版 权 所 有 


图 6.24 车 源 信息 查询 模块 的 运行 结果 
车 源 信息 查询 功能 的 实现 通过 car_select.php 文件 来 完成 ， 首 先 与 数据 库 建立 连接 ; 然后 创建 一 个 
form 表单 ， 将 客户 的 地 址 信息 提交 到 当前 页 ， 作 为 查询 的 条 件 ; 最 后 编写 PHP 脚本 语句 ， 以 表单 中 获 
取 的 地 点 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 符合 条 件 的 信息 ， 并 且 将 查询 出 
的 信息 显示 到 页 面 中 。 其 关键 代码 可 以 参考 本 章 例 程 02。 


6.6.4 ”单元 测试 
在 对 车 源 信息 查询 模块 进行 测试 的 过 程 中 ， 当 输入 指定 的 出 发 点 和 到 达 地 点 进行 提交 后 ， 出 现 如 
图 6.25 所 示 的 页 面 ， 提 示 程 序 出 错 ， 没 有 找到 符合 条 件 的 记录 。 


分 析 错 误 原因 : 从 弹出 的 错误 信息 来 看 ， 错 误 应 该 与 数据 库 有 关 。 下 面 从 数据 库 开 始 查 找 错 误 的 
原因 ， 首 先 确定 程序 中 使 用 的 数据 库 、 数 据 表 和 字段 的 书写 是 否 正确 ， 仔 细 核对 没有 发 现 问题 ; 然后 


中 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


确认 使 用 的 select 查询 语句 的 格式 是 否 正确 ,经 检查 发 现 , 在 编写 select 查询 语句 时 使 用 了 错误 的 语句 
格式 , 在 执行 like 模糊 查询 时 没有 将 参数 的 两 侧 加 上 单 引号 (car_road like %S$select1% and car road like 
%$select2%) ， 从 而 导致 程序 中 类 型 不 匹配 的 错误 。 其 错误 语句 和 正确 语句 的 对 比如 图 6.26 所 示 。 


EE ,ie en se oe vai ee orate racoues 


发 货 单 F:\webpage\Tsoft\vlel\car_select. php on line 31 
a 候 查 要 的 路 线 不 存在 ! 
i 查询 至 提交 
车 焉 中 码 ED 车 辆 关 还 便 用 日 志 ED 
客户 信息 着 理 
车 源 信息 溃 理 
修改 密码 


图 6.25 车 源 信息 查询 的 错误 运行 结果 


《?phl 


p 
if ($select1l==true || $select2==true) { 
$query="select * fron tb_car where car_road 
$result=nysql_query ($query) ; 
if (nysql_nun_rows ($result) >0) [ 
while ($nyrow=nysql_fetch_array ($result)) [ rw 
?> 正确 语句 错误 语句 


<?php 
like '%$select1%’ and car_road like '%$select2%’" 


if ($select1==true || $select2==true) { 
$query="select * from tb_car where car_road 
$result=nysql_query ($query) ; 

if (mysql_mun_rows ($result) >0) { 
while ($nyrow=nysql_fetch_array ($result)) { 


.> 
6.26 错误 语句 和 正确 语句 对 比 图 


正确 的 格式 为 car road like '%$selectl%' and car road like '%$select2%'， 或 者 car road like 
"9%6".$select1."%' and car_road like '%".$select2."%'。 修 改 后 保存 程序 ， 程 序 正常 运行 。 


全 注意 : 在 执行 SQL 语句 的 过 程 中 ， 传 入 SQL 语句 的 参数 为 字符 型 数据 ， 所 以 在 传 入 参数 的 两 侧 
应 该 加 上 单 引 号 ， 表 示 该 数据 为 字符 型 ， 否 则 程序 将 发 生 类 型 不 匹配 的 错误 。 


[加 说明: 在 PHP 中 的 字符 串 有 两 种 表示 格式 : 第 1 种 是 将 字符 囊 用 单 引号 括 起 来 ， 第 2 种 是 将 字符 
串 用 双 引 号 括 起 来 。 如 果 采 用 第 2 种 格式 ，PHP 变量 可 以 直接 写 在 该 字符 串 中 ， 系 统 不 会 
将 该 变量 看 作 是 字符 串 的 一 部 分 ， 而 是 将 其 看 作 PHP 的 合法 变量 。 


6.7 发 货 单 管理 模块 设计 


6.7.1 发 货 单 管理 模块 概述 


发 货 单 管理 模块 主要 包括 发 货 单 的 填写 、 发 货 单 查询 、 发 货 单 打印 和 发 货 单 删除 。 发 货 单 管理 模 
块 的 主要 功能 结构 如 图 6.27 所 示 。 
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图 6.27 发 货 单 管理 模块 的 功能 结构 图 


6.7.2 发 货 单 管理 模块 技术 分 析 


如 何 才 能 发 挥 函 数 的 最 大 作用 ? 设计 者 不 但 要 能 够 从 表面 理解 函数 的 功能 和 作用 ， 而 且 要 能 够 以 
一 种 发 散 的 思维 方式 去 考虑 问题 ， 不 要 被 其 本 身 的 功能 所 束缚 ， 要 将 函数 中 某 些 功能 有 机 地 结合 起 来 ， 
从 而 发 挥 其 更 大 的 作用 。 

在 发 货 单 管理 模块 中 应 用 到 一 个 发 货 单 的 编号 ， 该 编号 是 发 货 单 的 唯一 标识 ， 不 允许 重复 。 在 编 
写生 成 这 个 编号 的 程序 时 ， 首 选 方案 是 应 用 随机 函数 获取 随机 的 数字 作为 编号 ， 这 是 一 个 最 直接 的 做 
法 ， 但 是 考虑 到 发 货 单 编 号 不 但 具有 唯一 性 ， 而 且 还 要 使 其 具有 一 定 的 标志 性 ， 从 这 个 角度 考虑 使 用 
随机 函数 就 有 些 不 适合 ， 因 为 其 不 具备 一 定 的 标志 性 。 这 时 ， 如 果 应 用 date0 函 数 则 可 以 达到 上 述 所 说 
的 要 求 ， 不 但 编号 具有 唯一 性 ， 而 且 还 有 规律 可 循 。 

为 了 能 够 更 好 地 理解 这 种 发 散 思 维 的 编程 思想 ， 首 先 介绍 一 下 date0 函 数 。 语 法 格式 如 下 : 


string date ( string format , int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 字符 串 而 产生 的 字符 串 ， 其 中 的 参数 timestamp 是 可 选 
的 ， 默 认 值 为 titme0， 即 如 果 没 有 给 出 时 间 戳 则 使 用 本 地 当前 时 间 。 
date0 函 数 的 参数 format 的 格式 化 选项 如 表 6.1 所 示 。 


表 6.1 参数 format 的 格式 化 选项 

参数 说 了 明 
小 写 的 上 午 和 下 午 值 ， 返 回 值 am 或 pm 
大 写 的 上 午 和 下 午 值 ， 返 回 值 AM 或 PM 
Swatch Internet 标准 时 ， 返 回 值 000 一 999 
月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 01 一 31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 Mon 一 Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 January 一 December 

小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 1 一 12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 0 一 23 
小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 01 一 12 


To le lw i> | 


pla 
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续 表 
说 明 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 00 一 23 
有 前 导 零 的 分 钟 数 ， 返 回 值 00 一 59 
判断 是 否 为 夏令 时 ， 返 回 值 如 果 是 夏令 时 为 1， 否则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 1 一 31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 Sunday 到 Saturday 
判断 是 否 为 冰 年 ， 返 回 值 如 果 是 闲 年 为 1， 否则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 01 一 12 
3 个 字母 缩写 表示 的 月 份 ， 返 回 值 Jan 一 Dec 
数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 1 一 12 
与 格林 威 治 时 间 相 差 的 小 时 数 ， 例 如 0200 
RFC 822 格式 的 日 期 ， 例 如 Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 00 一 59 
每 月 天 数 后 面 的 英文 后 级 ，2 个 字符 ， 例 如 st，nd，rd 或 者 也 。 可 以 和 j 一 起 使 用 
指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
星期 中 的 第 几 天 ， 数 字 表示 ， 返 回 值 为 0 一 6 
ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
y 2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
z 年 份 中 的 第 几 天 ， 返 回 值 0 一 366 
时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 
-43200 一 43200 


时 
淡 


colol- ol | lo ls lzle cl ls 


“| 志 |z 


Zz 


名 注意 : 这 里 有 效 的 时 间 稚 的 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54~2038 年 1 月 
19 日 03:14:07 (此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 ) 。 在 Windows 系统 中 此 
范围 限制 为 从 1970 年 1 月 1 日 ~2038 年 1 月 19 日。 


避 技巧 要 应 用 发 散 的 思维 方式 ， 在 考虑 编程 方法 时 就 不 能 被 date0 函 数 自身 的 功能 所 限制 ，date0 
元 数 获取 的 时 间 不 只 可 以 用 来 记录 历史 中 的 菜 一 时 刻 ， 而 且 可 以 将 其 应 用 到 其 他 的 地 方 ， 
作为 一 个 有 规律 的 随机 数 。 

在 本 项 目 中 的 发 货 单 填 单 模块 中 就 是 应 用 date0 函 数 获取 当前 时 间 和 车辆 ID 作为 发 货 单 的 编号 。 
关键 代码 如 下 : 


<?php 

echo date("YmdHis"): // 获 取 系 统 的 当前 时 间 
echo S$ljid: // 获 取 车 辆 的 ID 

or 


其 运行 结果 如 图 6.28 所 示 。 
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车 源 信息 查询 


回执 发 货 单 确认 
发 货 单 查询 
客户 信息 管理 
车 源 信息 管理 
修改 密码 


车 主 电话 : | 


发 货 地 址 


图 6.28 发 货 单 编号 生成 的 运行 结果 


证 技巧 : 程序 开发 不 只 是 要 看 到 其 表面 的 功能 和 属性 ， 更 重要 的 是 将 某 些 功能 或 者 属性 有 机 地 结合 
起 来 ， 从 而 开发 出 更 有 意义 的 功能 ， 这 才 是 程序 员 所 要 追求 的 。 例 如 ， 在 学 习 字 符 串 中 的 
str_ireplace() 函 数 时 ， 不 但 要 知道 该 函数 的 功能 是 实现 字符 串 的 替换 功能 ， 而 且 要 学 会 如 何 
巧妙 地 运用 其 实现 更 好 的 功能 。 当 换个 角度 去 考虑 这 个 函数 的 功能 时 ， 对 这 个 函数 的 参数 
字符 串 进行 颜色 改变 。 这 时 实现 的 功能 就 不 只 是 简单 的 字符 串 替 换 ， 而 且 对 字符 串 实现 了 
描 红 的 功能 。 


6.7.3 发 货 单 填 单 的 实现 过 程 


国 ”发 货 单 填 单 模块 使 用 的 数据 表 : tb_customer、tb_car log、tb_shopping 
发 货 单 的 填写 是 客户 在 确定 物流 配送 路 线 以 后 填写 的 一 个 货物 配送 详细 信息 单据 ， 内 容 包括 发 货 
单 编号 、 车 牌号 码 、 车 主 电话 、 发 货 人 、 发 货 人 电话 、 发 货 地 址 、 付 款 方式 、 收 货 人 、 收 货 人 电话 、 
收 货 人 地 址 和 配送 说 明 。 发 货 单 填 单 的 运行 结果 如 图 6.29 所 示 。 


发 货 单 本 号 : |[2007122514362913 车 但 是 码 ; [古人 tes 车主 电话 ; 36504wwwoy 
发 货 人 : [ls* 发 贷 人 电话 : [13604wessss 付款 方式 : [发 偶合 获 到 
抠 抽 天 及 习 
货物 扩 述 : 
到 
区 
发 贷 地 址 习 
习 
收 基 人 : 攻 w，，，aA 活 : [God 
区 丙 市 
收 售 地 址 习 
习 
| 哆 货物 有 长 春 送 到 沈阳 ,往返 时 间 为 天 司 
说 明 : 
习 


ax| El CS 


图 6.29 发 货 单 填 单 的 运行 结果 


发 货 单 填 单 的 实现 通过 insert_dds.php 文 件 来 完成 , 首先 随机 生成 一 个 订单 编号 ,然后 创建 一 个 form 
表单 ， 最 后 将 表单 中 的 数据 提交 到 数据 库 中 。 在 insert_dds.php 文件 中 使 用 的 重要 表单 元 素 如 表 6.2 


a 
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所 示 。 
表 6.2 发 货 单 填 单 中 使 用 的 重要 表单 元 素 

名 称 | 元 素 类 型 重要 属性 含 义 
method="post" action="insert dd _ok.php" onSubmit="javascript: retum 

forml form 表单 
check d 

fahuo id text id="fahuo_id" value="<?php echo date("YmdHis"):?><?php echo $1jid:?>" 发 货 单 编号 

car number text value="<?php echo $myrow[car number]?>" 车 牌号 码 

car tel text value="<?php echo $myrow[tel]:?>" 车 主 电 话 
<option selected="selected"></option><option> 发 货 人 付款 </option> 

fahuo 全 select . ee 和 的 四 n 付款 方式 
<option> 第 三 方 付款 </option> 

car lo textarea Cols="65" rows="5" 说 明 

Submit submit value=" 提 交 " 提交 表单 


将 表单 中 提交 的 数据 存储 到 数据 库 是 通过 insert_dd_ok.php 文件 来 完成 的 , 在 该 文件 中 首先 连接 数 
据 库 ， 然 后 通过 preg_match0 函 数 判 断 表单 中 提交 的 电话 号 码 的 格式 是 否 正 确 ， 最 后 将 表单 中 提交 的 数 
据 存储 到 tb_shopping 表 中 ， 并 且 将 车 辆 使 用 的 说 明 提交 到 tb_car log 表 中 ， 将 客户 信息 提交 到 
tb_customer 表 中 。 关 键 代码 如 下 : 


例 程 03 代码 位 置 : 光盘 \TM\06\wilgl\insert_dd_ok.php 


<2php 

if($_SESSION[admin user]—true){ // 判 断 登 录 的 用 户 是 否 正确 
include("conn/conn.php"): // 连 接 数 据 库 
Sfahuo_time=date("Y-m-d H:i:s"): // 获 取 系统 当前 时 间 
if($Submit==true){ /如 果 提 交 表 单 ， 则 执行 下 面 的 语句 


/让 中 中 中 中 站 于 束 事 字 守 中 中 下 中 吉 站 村 下 可 本 让 洒洒 [表单 中 提交 的 电话 号 码 的 格式 是 否 正确 于 相让 中 让 相机 下 下 机 下 机 下 可 下 机 来 闻 员 下 下 本 $f 
if(preg_match("/^Qd{3}-)(d{8})S/ "Qad{4}-)0d{7))8I*Qd{4}-)0d{8})sI^Nd{11})S/",Scar_tel.Scountes)){ 
if(preg_match("/^(\d{3}-)0d{8))sI"0d{4}-)0d{7))8I"0d{4})0d{8))s/"Nd{11})8/",sfahuo_tel,Scountes)){ 


if(preg_match("/*(\d{3}-)0d{8})8/ 0d{4}-)0d{7)SI0d{4}-)0d{8))s/"Qd{11})8/",Sshouhuo_tel, $countes){ 
让 /中 站 中 让 让 中 于 站 站 于 让 中 吾 让 中 站 本 中 站 相 站 相 站 壮 表 单 提交 的 数据 添加 到 tb _shopping 表 中 宗 率 六 亲 率 六 六 半 冰 亲 半 素 亲 半 迷 末 半 半球 半 半 站 于 冰 要 
$query="insert into tb_shopping (car_number,car tel,fahuo id.fahuo_user， 
fahuo_tel,fahuo_address,fahuo_content,fahuo_time,fahuo fk.shouhuo_user.shouhuo_address,shouhuo _tel)values('$car_number', 
‘Scar _tel','$fahuo _id','$fahuo_user','$fahuo_tel','$fahuo_address'.'$fahuo_content','$fahuo_time','$fahuo_fk','$shouhuo_user','$sh 
ouhuo_address','$shouhuo_tel)":; 
Sresult=mysql_query($query): 
/和 这 素 兴 于 中叶 中 中 寂 涉 中 中 中 中 交 车辆 使 用 说 明 添加 到 tb_car_10g 表 中 发 中 / 
$querys="insert into 
tb_car log(car log.car number.log date.fahuo id)values('$car log','$car number','$fahuo_ time','$fahuo id)"; 
S$results=mysql query($querys): 
/本 二 来 于 于 环球 可 半 于 字 叶 半 可 中 站 字 半 于 半 站 闻 # 闪 客户 信息 提交 到 tb customer 表 中 本 本 本 本 本 本本 本 于 本 于 可 刘 于 于 于 池 于 于 池 于 于 池 于 计 / 
Squeryss="insert into tb_customer 
(customer user.customer tel.customer address)values('$fahuo user'.'$fahuo tel'.'$fahuo address")": 
Sresultss=mysql_query(Squeryss): 
echo "<script>alert(' 发 货 单 添加 成 功 !"):window.location.hre 信 "indexs.php?lmbs= 发 货 单 '</script>": 
}else{ 
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echo "<script>alert( 您 输入 的 收 货 人 的 电话 号 码 格式 不 正确 !):history-backO</scripf>": } 
人 "<script>alert( 您 输入 的 发 货 人 的 电话 号 码 格式 不 正确 !!1"):history.backQO</script>"; } 
和 "<script>alert( 您 输入 的 车 主 的 电话 号 码 格式 不 正确 !!"):history.backQ</script>"; } 
2 jelse{ // 判 断 用 户 是 否 正确 登录 ， 如 果 不 是 则 返回 登录 页 面 
echo "<script>alert(' 请 您 正确 登录 !"): window.location href~'index.php':</script>":} 
?> 


6.7.4 发 货 单 查询 的 实现 过 程 


国 ”发 货 单 查询 模块 使 用 的 数据 表 : tb_car log、tb_shopping 

发 货 单 查询 是 为 了 便于 对 发 货 单 进行 查找 以 及 处 理 而 设计 的 一 个 功能 ， 通 过 其 可 以 准确 地 查找 到 
指定 的 发 货 单 ， 并 且 还 设置 一 个 发 货 单 删除 的 功能 ， 可 以 对 已 经 失效 或 者 作废 的 发 货 单 进行 删除 。 其 
实现 的 原理 主要 是 通过 以 发 货 单 编号 为 条 件 的 精确 查找 或 者 以 发 货 人 姓名 为 条 件 的 模糊 查询 ， 然 后 将 
查询 的 结果 输出 到 页 面 中 。 运 行 结果 如 图 6.30 所 示 。 


发 货 单 查询 : [下 贡 香 编 寻 可 。 007122511053215 [提交 | 

发 货 单 编号 : |[2007122511053215 ”| 车 牌号 码 : || 十 A-***** 联系 电话 ;1360434+*** 
发 货 人 : | 区 > 电话 : [Er 付款 方式 : [第 三 方 何 款 到 
人 引述 :| 大 及 习 

发 地址 | 必要 习 

收 货 人 : | 邦 ** 收 货 人 电话 : 136x##*# 

中 当地 址 : | 下 网 - 
发疯 单 处 再。 | 

说 明 : | 讲 送 到 沈阳 ,往返 3 天 时 本 

间 4 = 
发 沼 单 


6.30 发 货 单 查询 的 运行 结果 


发 货 单 查询 的 实现 主要 通过 hwys.php 文件 完成 ,首先 与 数据 库 建 立 连接 ,然后 创建 一 个 表单 forml， 
通过 该 表单 来 提交 查询 的 条 件 ， 可 以 选择 精确 查找 〈 以 发 货 单 编号 为 条 件 ) 或 者 模糊 查询 (以 发 货 人 
姓名 为 条 件 ) ， 最 后 根据 form1 表单 中 提交 的 数据 ， 执 行 查询 语句 ， 从 数据 库 中 读 取出 符合 条 件 的 发 
货 单 的 内 容 。 程 序 的 关键 代码 如 下 : 

例 程 04 ”代码 位 置 : 光盘 \TM\06\wlgl\hwys.php 


<?php session_start(): include("conn/conn.php"): /初始 化 session 变 量 ， 连 接 数据 库 
if($_SESSION[admin_user]=—=true){ // 判 断 管 理 员 是 否 是 正确 登录 


?> 
<!-- 省 略 了 部 分 代码 -> 


<2php 
/二 二 让 站 证 让 于 可 站 中 站 站 * 济 | 和 当 form1 表 单 提交 的 值 为 发 货 单 编号 时 ， 执 行 下 面 的 语句 二 站 本 相 站 本 中 站 本 中 下 本 本 站 
这 S$selectl 一 "发 货 单 编 号 " and $select!=""){ 
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o 


Squery="select * from tb_shopping where fahuo id='$select"; 


Sresult=mysql query($query): /执行 查询 语句 
S$myrow=mysql fetch array(S$result): // 获 取 查 结果 
» 


/本 本 证 市 本事 本 本 本 本 本 中 本 本本 本 本 站 四 上] 果 form1 表 单 提 交 的 值 为 发 货 人 时 ， 执行 下 面 的 语句 * 本 六 订 本 六 亲本 亲 训 本 六 证 证 六 让 证 订 / 
这 $selectl 一 "发 货 人 " and $select!=""){ 
Squery="select * from tb_shopping where fahuo user='$select"™; 


Sresult=mysql_query($query): // 检 索 指 定 发 货 人 的 信息 
Smyrow=mysql fetch array(Sresult): 
?> 
/省 略 部 分 HTML 代 码 
<td colspan="4" bgcolor="#F FFFFF"><textarea name="car log"> 


Dv -> 
<?php 

$querys="select * from tb_car log where fahuo id-'Smyrow[ 包 huo id]": /从 车 辆 日 志 表 中 查找 指定 的 数据 

Sresults=mysql_query($querys); // 执 行 查询 语句 

Smyrows=mysql fetch_array($results): // 获 取 查 询 结果 

echo $myrows[car log]: // 输 出 车 辆 日 志 信息 

?> 

</textarea></td> 

到- 根据 发 货 单 的 人 DD 设置 一 个 发 货 单 删除 的 超 链 接 ----------------------------- -- > 

<td bgcolor="#FFFFFF"><a href="fhd_qr.php?delete=<?php echo Smyrow[id]:?>"> 删 除 发 货 单 </a></td> 


a /省 略 部 分 HTML 代 码 
<?php }else{ // 判 断 管理 员 是 否 正确 登录 ， 如 果 错 误 则 返回 到 登录 页 

echo "<script>alert(' 请 您 正确 登录 !"): window.location.href='index.php';</script>"; 
} 


Rs 


发 货 单 删除 的 操作 通过 fhd_qr.php 文件 来 完成 , 主要 根据 超 链 接 中 提供 的 发 货 单 ID 执行 删除 的 操 
关键 代码 如 下 : 


例 程 05 ”代码 位 置 : 光盘 \TM\06\Wwlgl\fhd_qr.php 


<?php session_start(): include("conn/conn .php"): // 初 始 化 session 变 量 ， 连 接 数 据 库 
这 $_ SESSION[admin_ user]—true){ 
这 S$delete 一 tmue){ /判断 提交 的 发 货 单 ID 是 否 为 真 
$query="delete from tb_shopping where id='$delete"™; // 根 据 发 货 单 ID 执 行 删除 发 货 单 的 操作 


Sresult=mysql_query(Squery): 
1 中 中 计 刘 中 中 让 让 让 中 中 让 本 相让 本 中 中 站 站 站 发 货 单 出 | 除 成 功 ， 弹出 提示 ， 并 返回 到 上 一 页 * 半 率 站 闲 素 弟 冰 六 亲 素 六 半 米 亲 半 六 末末 六 冰 */ 
echo "<script>alert(' 发 货 单 删除 成 功 !"):window.location.href="indexs.php?lmbs= 发 货 单 查询 ':</script>"; 
中 
jelse{ 
让 时 让 站 闪 站 村 站 站 村 站 站 站 相 相 站 中 站 中 丰 中 相 相 发 货 单 删除 失败 ， 弹出 提示 ， 并 返回 到 上 一 页 * 素 六 率 兴 率 闲 六 于 六 六 率 半 于 六 率 半 六 六 六 村 水 六 率 于 
echo "<script>alert( 请 您 正确 登录 !"): window.location.href~'index.php':</script>": 


=- 一 
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6.7.5 发 货 单 打印 的 实现 过 程 


发 货 单 的 打印 是 为 货物 的 发 送 提供 一 个 书面 的 依据 ， 作 为 企业 与 客户 商业 活动 的 凭证 ， 其 内 容 必 
须 真 实 、 准 确 。 发 货 单 打印 的 运行 结果 如 图 6.31 所 示 。 


ED 21¥ 


状 考 : 就 红 wie | 
各 注 Et | 
厂 打 印 广 梓 中 
页面 i 轩 
E71 仙 数 他 ): [| 


个 园 定 所 图 0】 个 当前 页 面 Q) 


Saari ws Lh 


图 6.31 发 货 单 打印 的 运行 结果 


这 里 的 发 货 单打 印 技术 主要 通过 框架 打印 技术 来 完成 ， 即 只 打印 框架 中 的 内 容 ， 而 网 页 中 其 他 的 
内 容 不 打印 。 实 现 原理 是 : 首先 在 insert_dd.php 文件 中 创建 一 个 浮动 框架 ， 设 置 要 指定 打印 内 容 的 范 
围 ,并 且 连 接 到 要 打印 的 insert_dds.php 文件 ,然后 在 指定 的 文件 中 输出 要 打印 的 内 容 , 最 后 应 用 onclick 
事件 调用 parent.content focus0 和 window.print() 方 法 实现 打印 功能 。 

在 insert_dd.php 文件 中 创建 一 个 浮动 框架 ， 设 置 框架 的 名 称 为 content， 设 置 打印 的 范围 宽 97%， 
高 252%， 连 接 到 要 打印 的 insert_dds.php 文件 ， 程 序 代码 如 下 : 

例 程 06 ”代码 位 置 ， 光盘 \TM\06\wlgl\insert_dd.php 


<iframe name="content" src="insert_dds.php?ljid=<?php echo Sljid:?>" frameborder="0" width="97%" 
height="252%"></iframe> 

设置 要 打印 的 内 容 ， 通 过 insert_dds.php 文件 来 完成 ， 发 货 单 填写 的 内 容 以 表单 的 形式 显示 ， 其 中 
的 内 容 可 以 参考 光盘 中 光盘 TM\06\wlgl\insert_dds.php 文件 的 相关 代码 ， 这 里 不 再 袭 述 。 

最 后 回 到 insert_dd.php 页 中 , 在 该 页 中 设置 一 个 超 链接 ,应 用 onClick 事件 调用 parent.content.focusO 
和 window.print() 方 法 实现 发 货 单 的 打印 功能 。 程 序 的 关键 代码 如 下 : 

例 程 07 ”代码 位 置 : 光盘 \TM\06\wlgl\insert_dd.php 

<a href="#" 

‘onClick="parent.content.focus():window.print():"> 
<img src="images/dl 033.gif' width="87" height="27" border="0"> 


</a> 
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6.7.6 单元 测试 


在 进行 发 货 单 填 单 模块 的 测试 过 程 中 ， 当 打开 发 货 单 填 单 模块 的 页 面 时 ， 其 运行 结果 如 图 6.32 
所 示 。 


EE 国 Fhian 
TRA 名 称 已 被 更 站 ， 或 者 暂时 不 可 用 。 
发 人 昌林 
客户 信息 管理 请 会 U 下 换 作 
本 。 吉 果 代 已 经 在 地 址 术 中 输入 该 阿 页 的 地 址 ， 请 确认 下拉 写 正确 - 
他 可 码 。 打开 ! 1E9_1 .se 主页 天后 查找 指向 络 克 兴 赴 信息 的 放 接 . 


。 单 击 人 ， 党 试 其 他 绕 接 。 
1 ME Ye om. 


JOTP 404 -~ 未 找到 文件 
Internet Explorer 


EE 
图 6.32 发 货 单 填 单 模块 的 运行 结果 


在 该 页 面 中 整个 浮动 框架 中 的 内 容 没 有 显示 。 分 析 错 误 原 因 ， 应 该 是 在 设置 浮动 框架 时 出 现 的 问 
题 ， 查 看 insert_dd.php 文件 中 的 代码 ， 发 现 原来 是 设置 浮动 框架 时 指定 的 链接 文件 出 现 了 问题 ， 使 用 
了 错误 的 链接 文件 名 称 。 链 接 文 件 的 正确 名 称 是 insert_dds.php， 而 程序 中 将 该 文件 的 名 称 书写 为 


inser_ddes.php， 从 而 导致 页 面 中 没有 输出 任何 的 内 容 。 将 insert_dd.php 文件 中 的 内 容 进 行 修改 ， 重 新 
运行 程序 ， 运 行 结果 如 图 6.33 所 示 。 


车 沽 信息 查询 发 贷 单 杭 号 : ||2007122515225513 | 车 牌 呈 码 : [再 -12545 车 主 电话 : [13604338784 
发 代入: | 发 各 人 电 活 : 从 天 方式 : 习 
回执 发 货 单 确认 
信贷 单 查询 人 物 扩 村 : 
客户 信息 管理 四 
车 涯 信息 基 理 a 
发 地址 
修改 密码 加 
收费 人 | 收 货 人 电话 : 
路 基地 址 : 于 
| 
(EE | 
说 明 : 
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到 到 
图 6.33 发 货 单 模块 的 运行 结果 
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6.8.1 


所 示 。 


6.8.2 


在 
里 应 用 


6.8 回执 单 验收 管理 模块 设计 


回执 单 模块 概述 


回执 单 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 。 将 该 发 货 单 执行 类 型 更 新 为 “1”， 表 明 该 次 物 


已 经 完成 ; 清空 该 车 辆 的 使 用 日 志 ， 便 于 执行 下 一 个 订单 。 该 模块 的 业务 操作 流程 如 图 6.34 
O 提交 货物 
回执 单 


6.34 ”回执 单 模块 的 业务 操作 流程 


回执 单 模 块 技术 分 析 


回执 单 模块 中 ， 要 对 回执 单 进行 确认 首先 要 查找 到 指定 的 发 货 单 ， 并 且 对 其 内 容 进行 核实 。 这 
的 是 根据 发 货 单 的 编号 进行 查询 ， 从 数据 库 中 读 取出 符合 条 件 的 数据 。 在 执行 查询 语句 、 从 数 


据 库 中 读 取 数据 时 ， 主 要 应 用 mysql_query0、mysql_num rows() 和 mysql_fetch_array0 函 数 。 

首先 通过 mysql_query0 函 数 执行 一 个 select 查询 语句 ， 然 后 通过 mysql_num_rowsO 函 数 判断 查询 
的 结果 是 否 有 值 ， 如 果 没 有 值 ， 则 输出 “您 查找 的 发 货 单 编号 不 存在 ! ”; 如 果 有 值 ， 则 执行 
mysql_fetch_arrau() 函 数 ， 输 出 数据 库 中 查询 到 的 数据 。 

mysql_query0 函 数 向 与 指定 的 连接 标识 符 关 联 的 服务 器 中 的 当前 活动 数据 库 发 送 一 条 MySQL 查 
询 ， 语 法 如 下 : 

Tesource mysql_query ( string query [, resource link identifier] ) 

mysql_query0 函 数 的 参数 query 是 字符 串 的 类 型 ， 传 入 的 是 SQL 指令 ， 参 数 link_identfier 是 资源 
类 型 ， 传 入 的 是 由 mysql_connectO 函 数 或 者 mysql_pconnect0 函 数 返 回 的 连接 号 。 如 果 省 略 该 参数 ， 则 
会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连 接 。 


会 注 


意 : 查询 字符 串 不 应 以 分 号 结束 。 


mysql_num rowsO 函 数 获取 结果 集中 行 的 数目 ， 语 法 如 下 : 


int 


mysql num rows (resource result ) 
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全 注意 : 该 函数 仅 对 SELECT 语句 有 效 。 

mysql_fetch_array() 函 数 返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false， 语 
法 如 下 : 

array mysql fetch array (resource result [. int result type] ) 

mysql fetch_array() 函 数 的 参数 说 明 如 表 6.3 所 示 。 

表 6.3 mysql_fetch_array() 函 数 的 参数 说 明 
说 了 明 

资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql query0 函 数 返回 的 “数据 指针 ” 
可 选项 ， 整 数 型 参数 ， 要 传 入 的 是 MYSQL ASSOC、MYSQL NUM 和 MYSQL BOTH 3 种 由 PHP 定 义 
好 的 常数 之 一 ， 默 认 值 是 MYSQL _ BOTH 
用 MYSQL ASSOC 只 得 到 关联 索引 (相当 于 mysql _fetch_assoc0 函 数 ) 
用 MYSQL NUM 只 得 到 数字 索引 〈 相 当 于 mysql fetch_row0 函 数 ) 
用 MYSQL _ BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 
全 注意 : 本 函数 返回 的 字段 名 是 区 分 大 小 写 的 。 

通过 上 述 3 个 函数 就 可 以 完成 从 数据 库 中 读 取 数 据 ， 并 且 将 数据 输出 到 页 面 的 功能 。 
公 注意 ; 使 用 该 种 方法 输出 的 只 是 数据 库 中 的 一 条 数据 。 如 果 加 上 while 循环 语句 ， 那 么 就 可 以 

完成 数据 库 中 所 有 符合 条 件数 据 的 循环 输出 。 

应 用 while 循环 语句 读 取 数 据 库 中 的 数据 ， 这 里 以 读 取 tb_shopping 表 中 的 数据 为 例 ， 程 序 代码 

如 下 了 


参数 名 称 


result 


Tesult_type 


<?php include("conn/conn.php"): // 连 接 数 据 库 
$query="select * from tb_shopping ": // 编 写 查询 语句 
Sresult=mysql_query($query): // 执 行 查询 语句 

© if(mysql num rows($result)<1){ // 判 断 数据 库 中 是 否 存 在 数据 
echo "您 查找 的 内 容 不 存在 !"; 
}else{ 

@ while($myrow=mysql_fetch_array(Sresulb){ /循环 输出 数据 库 中 的 数据 

?> 

<?php 

echo Smyrow[id]: // 输 出 数据 库 中 数据 

?> 

这 // 输 出 内 容 的 代码 部 分 略 

<?php }7> 

人 代码 贴 十 


@ mysql num rows0: 返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 
四 mysql fetch array0: 获取 结果 集中 行 的 数目 。 该 函数 只 对 select 语句 有 效 。 
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6.8.3 ”回执 单 验收 模块 的 实现 过 程 


国 。 回执 单 验收 模块 使 用 的 数据 表 : tb_car log、tb_shopping 
回执 单 验收 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 ， 运 行 结果 如 图 6.35 所 示 。 


车 沽 信息 查询 上 aE E33 
发 货 单 
[| 发 让 单 辆 号 :ooT122511053215 9: | [联系 电话 : [3507 
pr 2 和 人: [0 电话 : EE 付款 方式 [邦和 和 可 
客户 信息 管理 贡 物 据 述 : 开 | 
车 沽 信 息 管理 一 一 
修改 密码 
waAais: ED 
EEC 生 寺 执行 发 货 单 确认 的 操作 


6.35 ”回执 单 验收 模块 的 运行 结果 


回执 单 验收 模块 的 实现 过 程 是 : 首先 在 select_dhd.php 页 中 根据 发 货 单 的 编号 查询 出 指定 的 发 货 
单 ， 对 发 货 单 进行 核实 ， 然 后 单 击 “ 发 货 订单 确认 ” 超 链 接 ， 实 现 回 执 单 的 验收 。 关 键 代 码 如 下 : 


例 程 08 ”代码 位 置 : 光盘 \TMVWO6\vwlgl\select_ dhd.php 


<?php 
include("conn/conn.php"); // 连 接 数 据 库 
这 $Submit 一 true){ 
$query="select * from tb_shopping where fahuo id=-'Sselect": /编写 查询 语句 
Sresult=mysql_query(Squery): // 执 行 查询 语句 
if(mysql_num rows(S$result)<1){ 1/ 判断 数 据 库 中 是 否 存 在 数据 
echo "您 查找 的 发 货 单 编号 不 存在 1"; 
}else{ 
Smyrow=mysql_fetch_array(S$result): /输出 数据 库 中 的 数据 
> 
// 输 出 内 容 的 代码 部 分 略 


<td colspan="4" bgcolor="#FFFFFF"> 
<textarea name="car log"> 


?ph 
a 六 汪汪 可 于 可 于 半 尝 本 站 # 相 据 提交 的 货物 编号 ， 查 询 车 辆 日 志 中 的 对 应 日 志 兴 本 于 ## 站 和 / 
Squerys="select * from tb_car log where fahuo id=-'Smyrow[fahuo id]": 


Sresults=mysql query($querys): // 执 行 查询 语句 
S$myrows=mysql fetch amay(Sresults): /输出 数据 库 中 的 数据 
echo $myrows[car_ log]: /输出 数据 库 中 的 数据 
J 
</textarea></td> 
人 /输出 内 容 的 代码 部 分 略 
< 一- 一 一 一 一 一 一 一 一 一 一 设置 超 链接 ， 执 行 发 货 单 的 确认 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
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<td bgcolor="#FFFFFF"><a href="fhd_qr.php?fahuo id=<?php echo $myrow[fahuo id]:?>"> 发 货 订单 回执 确认 


</a></td> 
最 后 在 fhd_qr.php 页 根据 超 链接 提供 的 发 货 单 编号 更 新 发 货 单 类 型 为 “1”， 清空 对 应 车 辆 的 使 用 


日 志 。 其 程序 的 关键 代码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\06\wlgl\fhd_qrphp 
<?php session_start(); include("conn/conn.php"): // 初 始 化 session 变 量 
// 判 断 用 户 是 否 是 正确 登录 
// 判 断 超 链接 传递 的 变量 值 是 否 为 真 


if($_SESSION[admin user]—true){ 
if($_GET[fahuo idj==true){ 
/本 训 市 市 字 本 认 中 本 本 让 本 让 语 本 本 二 可 [站 站 如 钾 接 传 递 的 变量 值 为 条 件 ， 编写 删除 语 句 二 李 本 机 机 本 本 机 本 中 让 让 中 证 本 市 机 */ 


Squery="delete from tb_car log where fahuo id='$fahuo id'": 
S$result=mysql query(Squery): /执行 删除 语句 
/本 贞 宙 于 让 市 市 市 训 认 市 亨 证 证 本 让 本 本 让 本 划 亲 [发 货 单 数据 表 中 发 货 的 类 型 为 bad 1 史 本 率 率 束 闵 率 训 这 六 计 浆 闵 奈 冰冰 半球 冰冰 站 闪闪 闲 水 / 
$query="update tb_shopping set fahuo ys='1' where fahuo id='$fahuo id ": 
Sresult=mysql_query(Squery): 

echo "<script>alert(' 发 货 回 执 单 处 理 成 功 !):window.location href-'indexs.php?lmbs= 回 执 发 货 单 确认 


</script>": } 
}else{ 
echo "<script>alert( 请 您 正确 登录 !"); window.location href='index.php':</script>":; 


) 


?> 


已 和 


6.9 基础 信息 管理 模块 设计 


6.9.1 基础 信息 管理 模块 概述 


基础 信息 管理 模块 主要 包括 客户 信息 管理 ， 用 于 添加 客户 信息 和 删除 客户 信息 ; 车 源 信息 管理 ， 
主要 用 于 对 车 源 信息 进行 管理 ， 实 现 车 源 信息 的 添加 、 修 改 和 删除 功能 ， 管 理 员 信息 管理 ， 用 于 修改 


管理 员 的 密码 。 基 础 信息 管理 模块 的 功能 结构 如 图 6.36 所 示 。 


图 6.36 基础 信息 管理 模块 的 功能 结构 图 


310 


第 6 章 物流 配送 信息 网 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


6.9.2 ”基础 信息 管理 模块 技术 分 析 


修改 管理 员 密码 的 模块 中 ， 在 对 密码 进行 处 理 的 过 程 中 用 的 是 面向 对 象 技术 实现 密码 的 修改 。 

面向 对 象 技 术 的 关键 就 是 类 的 创建 ， 首 先 创建 一 个 chkuser 类 ， 其 中 定义 5 个 数据 成 员 ， 实 现 对 管 
理 员 密码 的 修改 ， 分 别 是 管理 员 名 称 〈$admin user) 、 管 理 员 密码 ($admin pass) 、 管 理 员 新 密码 
($admin new_pass) 、 管 理 员 新 密码 确认 〈$admin new_pass2) 和 验证 码 ($yzm) ， 然 后 应 用 chkuser 
函数 实现 数据 成 员 的 初始 化 ， 最 后 应 用 数据 成 员 函 数 chkinput 实现 密码 的 修改 。 程 序 代码 如 下 : 


例 程 10 ”代码 位 置 : 光盘 \TM\06\wlgl\update_pass_ok.php 


<2?php session start(); // 初 始 化 session 变 量 

© class chkuser{ // 创 建 一 个 chkuser 类 
Var $admin user: // 创 建 数 据 成 员 ， 用 户 名 
var Sadmin pass: // 创 建 数 据 成 员 ， 密 码 
Var $admin new_pass: /创建 数据 成 员 ， 新 密码 
Var $admin new_pass2: // 创 建 数据 成 员 ， 密 码 确认 
Var $yzm; /创建 数据 成 员 ， 验 证 码 

function chkuser($x,Sy.Sm.Sn,$z){ /创建 chkuser 函 数 ， 实 现 数据 成 员 初始 化 


Sthis->admin_ user=$x; 
Sthis->admin pass=$y; 
Sthis->admin_ new_ pass=S$m; 
Sthis->admin_new_ pass2=$n; 
Sthis->yzm=$z; 
@ function chkinputO{ // 创 建 chkinput 函 数 实现 密码 更 改 
// 判 断 输入 的 验证 码 是 否 正确 
© 这 strval($this->yzn)!=$_ SESSION["autonuml"]){ 
echo "<scriptf>alert( 验 证 码 输 入 错误 !):history.backO:</script>": 
exit; 
} 
include("conn/conn.php"): // 连 接 数 据 库 
// 判 断 输入 的 密码 是 否 正确 
$query=mysql query("select * from tb_admin where admin user=".S$this->admin user." and 
admin pass="".$this->admin pass.""): 
这 mysql_num rows(Squery)<D){ 
/ 坟 浊 于 二 本 池 二 李 汪 本 二 中 于 二 业 本 可 * 双 ] 果 密码 不 正确， 则 提示 重新 输 入 密码 字 本 站 汪汪 于 虽 宣 于 于 守 / 
echo "<script>alert( 您 输入 的 密码 不 正确 ， 请 重新 输入 ):history.backO:</script>": 
jelse{ 
/幸村 二 本 汪 二 本 汪 汪 汪汪 ## 加] 果 密码 正确， 则 应 用 更 新 语句 ， 实 现 密码 的 更 改革 # 站 本村 下 本 汪 呈 汪汪 烛 本 人 
$query1=mysql query("update tb_admin set admin pass="".$this->admin new_ pass.""): 
这 $query1 一 true){ 
echo "<script>alert( 密 码 更 改 成 功 ):history.backO:</script>": 
} 


} 


/二 可 刘 本 中 于 下 囊 训 李惠 本 宙 囊 宙 字 下 村 于 训 字 于 字 可 本 守 李 站 本 本 本 地 站 < 半 ] - 例 | 化 jj 开本 本 本 本 本事 本 本 检 本 本 事 本 本 机 可 本 于 本 字 于 本 于 字 束 下 可 宙 可 字 事 汪 事 刘 事 守 / 
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9 Schk=new 
chkuser($ POST[admin user].md5($_ POST[admin pass]).md5($ POST[admin new pass]).$ POST[admin new_ pass2].$ P 
OST[yzm]): 
S$chk->chkinputO: 
?> 

< 代码 贴 填 

@ chkuser: 创建 一 个 类 名 称 为 chkuser， 用 于 修改 管理 员 密 码 。 

@ chkinput 0: 创建 一 个 自 定义 函数 chkinputO0， 用 于 实现 密码 修改 。 

@ strval 0: 获取 变量 的 字符 囊 。 

@ new chkuser: 实现 类 的 实例 化 ， 应 用 类 实现 密码 修改 。 


6.9.3 客户 信息 管理 的 实现 过 程 


国 ”客户 信息 管理 模块 使 用 的 数据 表 : tb_customer 
客户 信息 管理 模块 的 主要 功能 就 是 向 数据 库 中 添加 客户 信息 ， 并 且 可 以 对 指定 的 客户 进行 删除 操 
作 。 客 户 信息 管理 模块 的 运行 结果 如 图 6.37 所 示 。 


客户 信息 管理 
gpPkta: | 
电话 : 1360433ss** 
联系 地 址 | 过 市 习 
提交 | 

客户 姓名 电话 联系 地 址 换 作 
Ne 0431-12345676 长 春 市 MR 
En 13804338784 四 市 pr 


图 6.37 客户 信息 管理 模块 的 运行 结果 


客户 信息 管理 模块 主要 通过 customer.php、customer_ok.php 和 delete_customer.php 文件 来 完成 。 首 
先 通过 customer.php 文件 来 添加 客户 信息 ， 并 且 输 出 数据 库 中 的 所 有 客户 信息 ， 设 置 删除 客户 信息 的 
超 链接 。 其 输出 客户 详细 信息 的 关键 代码 如 下 : 


例 程 11 代码 位 置 : 光盘 \TM\06\wlgl\customer.php 


<?php include("conn/conn .php"): // 连 接 数 据 库 
S$query=mysql_query("select * from tb_customer "): // 执 行 查询 语句 ， 从 数据 库 中 读 取 数据 
while($myrow=mysql_fetch_array(Squery)){ // 通 过 while 循 环 语句 ， 循 环 输出 数据 库 中 的 数据 
?> 
<tr> 


<td height="26" align="center" bgcolor="#FFFFFF"><?php echo Smyrow[customer user]:?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo Smyrow[customer tell:?></td> 

<td align="center" bgcolor="#FFFFFF"><?php echo Smyrow[customer address]:?></td> 
i -设置 删除 客户 信息 的 超 链接 ， 并 且 设置 客户 的 ID 为 变量 一- > 
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<td align="center" bgcolor="#FFFFFF"><a hre 全 "delete_customerphp?delete=<?php echo 
$myrow[customer id]:?>"> 删 除 </a></td> 
<t> 
<?php }7> 


客户 信息 添加 的 操作 通过 customer_ok.php 文件 完成 ， 首 先 连接 数据 库 ， 然 后 判断 提交 的 电话 号 码 
的 格式 是 否 正确 ， 最 后 将 客户 信息 数据 提交 到 数据 库 中 。 程 序 的 代码 如 下 : 


例 程 12 ”代码 位 置 : 光盘 \TM\06\wlgl\customer_ok.php 
<?php include("conn/conn.php"): // 连 接 数 据 库 
if($Submit==true and $customer user—true){ // 判 断 提 交 的 值 是 否 为 真 
/本 李 认 证 市 市 训 市 本 宙 亨 训 市 训 主 本 本 亨 本 本 本 本 本 让 二 可 本 内 | 用 [ 提 交 的 电话 号 码 格式 是 否 正确 可 让 可 机 中 机 本 相机 机 本 机 训 环 本 训 可 本 市 事 机 市 事 市 市 束 */ 
if(preg_match("/*(N\d{3}-)(\d{8})SIQd{4}-)0d{7})sI Qad{4}-)0d{8})sI*Qd{11})3/".Scustomer tel,$countes)){ 
/本 机 语 市 市 本 市 市 本 机 字 证 本 本 计 本 认 让 让 本 机 二 说 本 本 二 二 痊 提交 的 数据 添加 到 客户 信息 表 中 池 相 机 相机 本 机 本 让 机 本 刘 本 本 市 事 机 市 事 本事 束 */ 


$query=mysql_ query("insert into 
tb_customer(customer user,customer tel,customer address)values('$customer User.Scustomer tel','$customer address")"); 
这 $query 一 true){ /如 果 添 加 操作 成 功 ， 弹 出 提示 ， 并 返回 首页 
echo "<script> alert( 客 户 信息 添加 成 功 ):window.location href-'indexs.php?lmbs= 客 户 信息 管理 
</script>"; 
y 
Jelse{ // 如 果 添加 操作 失败 ， 弹 出 提示 ， 并 返回 上 一 页 
echo "<script>alert( 您 输入 的 电话 号 码 格式 不 正确 !!"):history.backQO</script>"; 
} 
1 
?> 


客户 信息 删除 的 操作 通过 delete_customer.php 文件 来 完成 ， 主 要 以 超 链接 中 提交 的 变量 为 根据 ， 
删除 数据 库 中 指定 的 客户 信息 。 程 序 的 关键 代码 如 下 : 


例 程 13 ”代码 位 置 ; 光盘 \TM\06\wlgl\delete_customer.php 
<?php include("conn/conn.php"): // 连 接 数 据 库 
这 $delete 一 tme){ /独断 提 交 的 变量 值 是 否 为 真 

人 中 让 让 中 中 于 中 中 让 让 中 让 中 中 本 站 本本 中 站 相 撕 SGET 获 取 的 变量 值 为 依据 | 执行 删除 语句 于 让 站 村 中 中 机 字 下 由 事 事 下 下 事 站 下 可 吕 bi 

$query=mysql_ query("delete from tb_customer where customer id='$ GET[delete]”): 

这 Squery){ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 ， 并 返回 首页 

echo "<script> alert( 客户 信息 添加 成 功 "):window.location.href='indexs.php?lmbs= 客 户 信息 管理 
"</script>"; 


6.9.4 车 源 信 息 管理 的 实现 过 程 
国 车 源 信息 管理 模块 使 用 的 数据 表 : tb_car 


车 源 信息 管理 模块 主要 实现 对 车 辆 信息 的 添加 、 修 改 和 删除 操作 。 车 源 信息 管理 模块 的 运行 结果 
如 图 6.38 所 示 。 
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车 天 信息 查询 
发 货 单 车 源 信息 管理 二 4-78745 司 提交 
回执 发 上 音 确 认 二 
发 侨 单 查询 闫 名 : [Fis* ib [Em 
客户 信息 管理 身份 证 号 : |[220322821eees* | 1 = 
店铺 拉 述 : 
电话 : 。 [60m 
修改 密码 | 是 
春 - 沈 阳 -大 连 2 
地 人 a - 运 多 中 二 : 
到 


图 6.38 车 源 信息 管理 模块 的 运行 结果 


车 源 信息 管理 模块 主要 由 两 个 文件 组 成 。 一 个 是 car.php 文件 ， 以 表单 文件 为 主 用 于 输出 车 辆 的 信 
息 和 提交 车 辆 的 信息 。 设 置 form2 表单 ， 根 据 车 牌号 码 从 数据 库 中 读 取 对 应 车 辆 的 信息 ， 进 而 实现 对 
该 车 辆 信息 的 修改 或 者 删除 操作 ;设置 forml 表单 ， 用 于 显示 从 数据 库 中 读 取 的 车 辆 信息 和 直接 添加 
车 辆 信息 到 数据 库 。forml 表单 中 使 用 的 重要 元 素 如 表 6.4 所 示 。 


表 6.4 车 源 信息 管理 模块 中 使 用 的 重要 表单 元 素 


1 
名 称 元 素 类 型 重要 属性 含义 
method="post" action="car insert_ ok.php” 表单 


forml form ew . 
onSubmit="javascript:return check_car(forml):” 


Usermame text id="usemame" value="<?php echo $myrows[usermame]:?>" 车 主 姓名 


car_number text id="car_ number" value="<?php echo $myrows[car_number]:?>" 车 牌号 码 
car tel text Value="<?php echo Smyrow[tel]:?>" 车 主 电话 


user_number text id="user_number" value="<?php echo $myrows[user_number]:?>" 身份 证 号 码 
car_content textarea <?php echo Smyrows[car_content]:?> 车 辆 描述 
address textarea <?php echo Smyrows[address]:?> 家 庭 地 址 
car road textarea <?php echo Smyrows[car road]:?> 运输 路 线 
Submit submit value=" 提 交 " 提交 表单 
Submit2 submit Value=" 修 改 " 提交 表单 
Submit4 Submit value= "删除 " 提交 表单 


另 一 个 是 car_insert_ok.php 文件 , 通过 该 文件 实现 对 表单 中 提交 的 数据 进行 处 理 。 首 先 连接 数据 库 ， 
然后 根据 表单 中 提交 的 值 进行 判断 ， 当 按钮 SSubmit 的 值 为 “提交 ”时 ， 将 提交 的 数据 存储 到 指定 的 数 
据 表 中 ; 当 按 钮 SSubmit2 的 值 为 “修改 ”时 ， 则 更 新 语句 ， 对 车 辆 的 信息 进行 更 新 ; 当 按 钮 SSubmit4 
的 值 为 “删除 ”时 ， 则 执行 删除 语句 ， 将 指定 的 车 辆 信息 删除 。 程 序 的 关键 代码 如 下 : 


例 程 14 ”代码 位 置 光盘 \TMVWO6Vvwlglvcar insert_ ok php 


<?php session_start(): include("conn/conn.php"): // 初 始 化 session 变 量 ， 连 接 数 据 库 
© if($_SESSION[admin user]—true){ // 判 断 管 理 员 是 否 是 正常 登录 
if($Submit 一 "提交 "){ // 判 断 表单 提交 的 值 是 否 为 "提交 " 
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/二 二 池 本 出 二 示 池 本 业 站 直 汪汪 站 本寺 # 潮 | 漠 [ 表 单 中 提交 的 电话 号 码 的 格式 是 否 正确 兴 字 二 兴 * 相 汪汪 机 中兴 洒 机 册 洒 本 
四 这 preg_ match("Adf117 IdX]IMd1S} .Suser numberScounts)){ 

这 preg match(wGd3}-Qdt8)SACd-JGd7DSIAQdC-)GQdf8)DSIACdf119)S/Suser tel.Scountes)){ 

/二 站 下 站 下 囊 宙 下 地 和 直 站 ## 闪 表单 中 的 数据 添加 到 jtb_car 数 据 表 中 于 

$query=mysql query("insert into 
tb_car(usemame,user number,tel,address,car number,car road.car content)values('$username','$user number','$user tel','$addr 
ess','$car number','$car road','$car content)"); 

if($query—true){ 
echo "<script>alert(' 车 源 信息 添 加 成 功 !"):window.location.href='indexs.php?lmbs= 车 源 信息 管 


理 ':</script>": } 
}else{ 
echo "<script>alert( 您 输入 的 电话 号 码 格式 不 正确 !1"):history.backQ 〇 </script>":  } 
}else{ 
echo "<script>alert( 您 输入 的 身份 证 号 码 的 格式 不 正确 !!"):history.backQ</script>": 。 } 
lb 
/让 中 让 让 中 市 让 中 市 下 中 本 本 让 机 中 中 让 本 本 下 二 可 语 二 本 本 本本 || 肝 和 夫 单 提交 的 值 是 否 为 "修改 "中 机 机 本 机 让 机 本 让 让 本 让 训 机 机 机 语 中 机 本 员 市 本 让 事 机 */ 
@ if(SSubmit2 一 "修改 "){ 


/二 林寺 于 于 于 来 于 于 吾 本 于 六 灶 于 刘 半 本 汪汪 于 千村 本 汪 洁 | 计 [ 电 话 号 码 的 格 式 是 否 正确 下 守 于 六 来 素 半 率 六 站 半 冰 六 六 / 
if(preg_match("N\d{17}MNd[X]Nd{15}/",$user number.Scounts)){ 
if(preg_match("/*(\d{3}-)0d{8})s/ "0d{4}-)0d{7))s/0d{4}-)0d{8}) /0d{11))s/",Suser_tel,Scountes)){ 
/* De 1 5 二 : 中 的 数据 理 刘 让 刘 让 中 市 让 市 市 让 本 证 市 刘 于 刘 训 本 市 训 训 事 束 训 */ 
$query="update tb_car set username='$username', user number='$user number', tel='$user tel', 
address='$address', car_number='$car number', car road='$car road', car_content='$car content"™; 
Sresult=mysql_query($query); 
if($result==true){ 
echo "<script>alert( 车 源 数据 更 新 成 功 !):window.location href-'indexs.php?lmbs= 车 源 信息 管 
理 ':</script>";} 
}else{ 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !):history.backO</script>":} 
}else{ 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 !1"):history.backQ</script>"; } 


} 
/中 半 半 素 半 六 宗 守 六 宗 半 站 素 站 半 半 站 站 半球 中 守 相 中 站 中 相关 | 靳 表 单 提 交 的 值 是 否 为 "出 | 除 "于 半球 下来 于 六 素 六 闪 守 半 半 家中 半球 站 半球 站 半 宗 站 半 / 


9 if(SSubmit4 一 "删除 "){ 
Squery="delete from tb_car where car_ number=-'$car_ number": 1/ 删除 tb_car 数 据 表 中 的 数据 
$result=mysql_query($query); 
if($result=—=true){ 
echo "<script>alert(' 车 源 数 据 删 除 成 功 !"):window.location.href='indexs.php?lmbs= 车 源 信息 管 
理 ':</script>": 
上 
} 
?> 
<2php }else{ 
echo "<script>alert( 请 您 正确 登录 !"):; window.location.href~'index.php':</script>":; 
} 
?> 
人 代码 贴 十 


@ $_SESSION[admin user]: 判断 用 户 是 否 是 正常 登录 。 
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@ preg_match0: 应 用 preg match 验证 电话 号 码 的 格式 是 否 正 确 。preg match(0 对 正则 表达 式 进行 匹配 。 返 回 匹配 
的 次 数 ，0 次 (没有 匹配 ) 或 者 1 次 ， 因 为 preg_match() 函 数 在 第 一 次 匹配 之 后 将 停止 搜索 ， 出 错 返回 false。 

罩 $Submit2 一 "修改 ": 判断 表单 提交 的 值 是 否 等 于 修改 ， 如 果 等 于 修改 则 执行 修改 的 操作 。 

@ $Submit4 一 "删除 ": 判断 表单 提交 的 值 是 否 等 于 删除 ， 如 果 等 于 删除 则 执行 删除 语句 。 


6.10 开发 技巧 与 难点 分 析 


6.10.1 应 用 存储 过 程 实现 管理 员 登 录 


在 物流 配送 信息 网 系统 中 ， 管 理 员 登 录 是 通过 存储 过 程 来 实现 的 。 下 面 就 来 讲解 一 下 如 何 创建 和 
应 用 存储 过 程 。 
MySQL 5.0 以 后 的 版 本 中 开始 支持 存储 过 程 ， 存 储 过 程 具 有 一 致 性 、 高 效 性 、 安 全 性 和 体系 结构 
等 特点 ， 其 将 在 以 后 的 项 目 中 得 到 广泛 的 应 用 。 
一 个 完整 的 存储 过 程 包括 名 称 、 参 数列 表 ， 以 及 很 多 SQL 语句 集 。 下 面 以 一 个 存储 过 程 的 定义 过 
程 为 例 ， 对 其 结构 进行 详细 的 讲解 。 
©@ createprocedure procedure name (in parameter integer) 
四 begin 
declare variable varchar(20): 
if parameter=] then 
set variable='Apache’; 
else 
set variable="MySQL'; 
end 证 
insert into tb_table (tb_name) values (variable): 
© end: 
人 代码 贴 二 
@ create procedure: 创建 存储 过 程 ， 以 关键 字 create procedure 开始 ， 后 面 紧 跟 存储 过 程 的 名 称 和 参数 。 
@ begin: MySQL 存储 过 程 的 语句 块 以 begin 开始 。 
@@ end: MySQL 存储 过 程 以 end 结束 。 


MySQL 中 存储 过 程 的 建立 以 关键 字 create procedure 开始 ， 后 面 紧 跟 存 储 过 程 的 名 称 和 参数 。 
MySQL 的 存储 过 程 名 称 不 区 分 大 小 写 ， 例 如 PROCEDURESO 和 procedures() 代 表 同 一 存储 过 程 名 。 存 
储 过 程 名 不 能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 

存储 过 程 的 参数 一 般 由 3 部 分 组 成 。 第 1 部 分 可 以 是 in、out 或 inout。in 表示 向 存储 过 程 中 传 入 
参数 ，out 表示 向 外 传 出 参数 ; inout 表示 定义 的 参数 可 传 入 存储 过 程 并 可 以 被 存储 过 程 修改 后 传 出 存 
储 过 程 ， 存 储 过 程 默 认为 传 入 参数 ， 所 以 参数 in 可 以 省 略 。 第 2 部 分 为 参数 名 。 第 3 部 分 为 参数 的 类 型 ， 
该 类 型 为 MySQL 数据 库 中 所 有 可 用 的 字段 类 型 ， 如 果 有 多 个 参数 ， 参 数 之 间 可 以 用 去 号 进行 分 割 。 

MySQL 存储 过 程 的 语句 块 以 begin 开始， 以 end 结束 。 语 句 体 中 可 以 包含 变量 的 声明 、 控 制 语句 、 
SQL 查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 应 将 语句 结束 标志 “:” 
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更 该 为 其 他 字符 , 并 且 该 字符 在 存储 过 程 中 出 现 的 几率 也 应 该 较 低 ， 可 以 用 关键 字 delimiter 更 改 , 例如: 


mysql>delimiter // 


delimiter 关键 字 之 间 保 留 一 个 空格 ， 否 则 该 存储 过 程 将 会 出 现 错误 ， 不 能 成 功 创建 。 
存储 过 程 创 建 之 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 procedure_name 指 存储 过 程 名 。 
drop procedure procedure_ name 
MySQL 存储 过 程 是 在 命令 提示 符 下 创建 的 ， 其 操作 步骤 如 下 : 
(1) 打开 “命令 提示 符 ”。 


全 注意 ; 在 应 用 delimiter 关键 字 更 改 存储 过 程 中 的 语句 结束 标志 “:” 时 ， 所 使 用 的 字符 必须 与 


(2) 进入 “命令 提示 符 ” 后 ， 首 先 登 录 MySQL 数据 库 服 务 器 ， 在 命令 提示 符 下 输入 如 下 命令 : 


mysql -u 用 户 名 -Pp 用 户 密码 

(3) 更 改 语句 结束 符号 ， 将 语句 结束 符 更 改 为 “//”， 代 码 如 下 : 
delimiter // 

(4) 创建 存储 过 程 前 应 首先 选择 某 个 数据 库 ， 代 码 如 下 : 

use 数据 库 名 

(5) 创建 存储 过 程 。 

(6) 通过 call 语句 调用 存储 过 程 。 


下 面 以 本 项 目 中 的 管理 员 登 录 模块 为 例 ， 对 存储 过 程 进行 实例 化 的 讲解 。 首 先 创建 存储 过 程 ， 在 


命令 提示 符 中 输入 代码 ， 如 图 6.39 所 示 。 
了 6 


jauery OK. @ rows affected (8.88 sec) 


图 6.39 创建 存储 过 程 
然后 在 index.php 文件 中 编写 PHP 脚本 语句 ， 调 用 存储 过 程 实现 用 户 登 录 。 关 键 代 码 如 下 : 
例 程 15 ”代码 位 置 : 光盘 \TM\06\wlgl\index.php 
<?php 


if($_POST[admin user]!="" || $ POST[admin pass]!=""){ /获取 表单 提交 的 数据 
$conn=new mysqli("localhost"."root"."root"."db wilgl"): /连接 数据 库 
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S$conn->query("set names gb2312"): /指定 字符 编码 格式 
S$admin user=$ POST[admin user]: /获取 变量 值 
Sadmin pass=md5($ POST[admin pass]): /获取 变量 值 
$sql=$conn->query("call admin reg(".$admin user.",".$admin pass.")"): // 调 用 存储 过 程 
Sres=$sql->fetch array(MYSQL BOTH): /输出 结果 
if($res!=NULD){ 

session register("admin user"): /定义 session 变 量 


session register("admin pass"); 

echo "<script>alert( 用 户 注册 成 功 !"):window.location.href="indexs.php':</script>"; 
jelse{ 

echo "<script>alert( 用 户 注册 失败 !"):</script>"; 
} 


6.10.2 ”应 用 正则 表达 式 验证 电话 号 码 


在 物流 配送 信息 网 中 ， 确 认输 入 的 电话 号 码 正确 与 否 是 一 个 非常 重要 的 问题 ， 如 果 客 户 在 填写 订 
单 时 使 用 了 错误 的 电话 号 码 ， 那 么 就 会 带 来 很 多 不 必要 的 麻烦 ， 所 以 在 填写 电话 号 码 时 一 定 要 注意 电 
话 号 码 的 准确 性 。 为 了 更 好 地 确保 电话 号 码 输入 的 准确 性 ， 可 以 通过 程序 对 其 进行 控制 。 虽 然 程序 不 
能 判断 电话 号 码 输入 的 是 否 正确 ， 但 是 通过 程序 可 以 对 电话 号 码 的 格式 进行 判断 ， 从 而 避免 一 些 电话 
号 码 在 录入 的 过 程 中 出 现 多 写 或 者 漏 写 数 字 的 错误 。 

验证 电话 号 码 格式 是 否 正 确 可 以 通过 正则 表达 式 来 完成 。 

在 进行 判断 之 前 首先 要 对 电话 号 码 的 格式 进行 分 类 : 第 1 类 区 号 是 3 位 数字 ， 其 他 是 8 位 数字 ; 
第 2 类 区 号 是 4 位 数字 ， 其 他 是 7 位 数字 ， 第 3 类 区 号 是 4 位 数字 ， 其 他 是 8 位 数字 ; 第 4 类 是 手机 
电话 号 码 ， 由 11 位 数字 组 成 。 

然后 编写 一 个 判断 电话 号 码 格式 的 正则 表达 式 ， 其 代码 如 下 : 

0d{3}-)Qd{8))5I "0d{4}-)0d{7))sI Nd{4}-)0d{8})s Nd{11})s/ 

正则 表达 式 的 功能 分 析 如 下 : 使用“^” 和 “$” 对 字符 串 进 行 边界 限制 ， 对 区 号 从 字符 串 的 开始 
进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “0” 中 的 内 容 作为 一 个 原子 使 用 ;使 用 
“\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ， 或 者 直接 判断 为 11 个 数字 ;使 
用 “{}” 来 对 前 字符 进行 重复 匹配 ， 使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 4 个 模式 。 

正则 表达 式 创建 完成 后 就 可 以 应 用 到 程序 中 对 电话 号 码 进行 判断 ， 要 完成 对 电话 号 码 的 判断 还 要 
应 用 到 一 个 preg_match0 函 数 。 

preg_match() 函 数 对 正则 表达 式 进行 匹配 。 返 回 匹配 的 次 数 ，0 次 (没有 匹配 ) 或 者 1 次 ， 如 果 出 
错 则 返回 false。preg_match() 函 数 的 参数 说 明 如 表 6.5 所 示 。 


表 6.5 preg_match() 函 数 的 参数 说 明 


必要 参数 ， 指 定 匹 配 的 正则 表达 式 
必要 参数 ， 指 定 要 搜索 的 字符 串 
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续 表 
说 明 
可 选 参数 ， 如 果 提 供 了 subpatterms， 则 其 会 被 搜索 的 结果 所 填充 。$ subpatterns [0] 将 包含 与 整个 模式 匹 
配 的 文本 ，$ subpattems [1] 将 包含 与 第 一 个 捕获 的 括号 中 的 子 模式 所 匹配 的 文本 ， 以 此 类 推 
可 选 参数 ,如果 设 定 标记 PREG OFFSET CAPTURE， 则 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 
串 偏 移 量 。 注意 这 改变 了 返回 的 数组 的 值 , 使 其 中 的 每 个 单元 也 是 一 个 数组 , 其 中 第 1 项 为 匹配 字符 串 ， 
第 2 项 为 其 偏 移 量 。 本 参数 自 PHP 4.3.0 开始 使 用 


参数 


subpatterns 


flags 


最 后 将 正则 表达 式 和 preg_match() 函 数 应 用 到 实际 的 程序 中 , 实现 对 电话 号 码 格式 的 判断 。 这 里 以 


car_insert_ok.php 文件 中 的 代码 为 例 ， 关 键 代 码 如 下 : 
例 程 16 ”代码 位 置 光盘 \TM\06\wlgl\car_insert_ok.php 
<?php 
这 SSubmit==tmue){ // 浏 断 表单 提交 的 值 
/站 二 直下 本 本 和 * 有 用 Dreg_Imatch 函 数 和 正则 表达 式 判断 身份 证 号 码 格式 是 否 正确 *########s##s# 本 天 
这 preg_match("Ad{f17}[WdIX]Md115}.Suser_ number,$counts)){ 
/本 站 半生 站 由 于 站 半生 半 站 站 中 有 1 用 DIeg_Iatch 函 数 和 正则 表达 式 判 断 电话 号 码 格式 是 否 正 确 闻 地 让 本 让 市 亨 市 市 证 市 本 可可 率 事 人 
iftpreg_match("/^(\d{3}-)(d{8})s/0d{4}-)0d{7))s/0d{4}-)0d{8))s/ 0d{11})s/",Suser tel scountes)){ 
广 *+yyytyty 如 果 格 式 正确 ， 则 执行 下 面 的 插入 语句 ， 将 车 辆 信息 添加 到 数据 库 中 ， 否 则 *yysywysss 4/ 
$query=mysql_query("insert into 
tb_car(usermame,user_ number,tel,address,car_number,car Toad.car_content)values(S$username'S$user_ number,S$user tel','$addr 
ess','$car number','$car road','$car content)"):; 
if(squery—true){ 
echo "<script>alert( 车 源 信息 添加 成 功 !):window.location href-'indexs.php?lmbs= 车 源 信息 管理 
‘</script>"; } 
j}else{ 
echo "<scripf>alert( 您 输入 的 电话 号 码 格式 不 正确 !1):history.backO</scripf>": } 
eet hn 上 出 提示 信息 ， 返回 到 表单 提交 页 ** 玉 直下 玉 于 于 素 六 于 率 六 站 半球 站 半球 半 闪 率 浆 */ 
echo "<script>alert( 您 输入 的 身份 证 号 码 的 格式 不 正确 !!"):history.backO 〇 </script>"; } 
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6.11 报表 打印 技术 


报表 打印 技术 的 实现 方法 很 多 ， 可 以 分 别 通过 Web、Word 打印 、Excel 和 CSS 样式 打印 等 。 在 本 
项 目 中 应 用 的 打印 方式 是 通过 Web 打印 ， 并 且 是 打印 指定 框架 中 的 内 容 ， 其 详细 的 讲解 可 参考 6.6.5 
节 发 货 单 打印 的 实现 过 程 。 下 面 再 介绍 一 种 通过 CSS 样式 来 打印 页 面 中 指定 内 容 的 方法 。 

通过 CSS 样式 打印 页 面 中 指定 内 容 主要 会 用 CSS 样式 的 media 类 型 。media 类 型 是 CSS 属性 媒体 
类 型 ， 用 于 直接 引入 媒体 的 属性 。 语 法 格式 如 下 : 


(@media screen | print | projection | braille | aural | tv | handheld | all 
media 类 型 的 参数 说 明 如 表 6.6 所 示 。 
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表 6.6 ”media 类 型 的 参数 说 明 


参数 说 明 

screen 默认 值 ， 指 提交 到 计算 机 屏幕 
print 指 输出 到 打印 机 

projection 指 提交 到 投影 机 

braille 提交 到 凸 字 触 觉 感 知 设备 
aural 指 语音 电子 合成 器 

tv 电视 类 型 的 媒体 

handheld 指 手持 式 显示 设备 

all 用 于 所 有 媒体 


个 注意 : 应 用 CSS 样式 打印 ， 控 制 的 是 指定 表格 的 背景 ， 只 是 将 表格 的 背景 颜色 进行 隐藏 ， 而 不 能 
控制 表格 中 内 容 的 输出 。 所 以 在 应 用 CSS 样式 进行 指定 内 容 的 打印 时 ， 前 提 是 不 打印 的 内 
容 必 须 是 以 表格 中 背景 图 象 的 形式 显示 的 ， 否 则 应 用 CSS 样式 进行 指定 内 容 打印 是 没有 效 
果 的 。 


这 里 以 项 目 中 的 客户 信息 为 例 ， 创 建 一 个 css_print.php 文件 ， 讲 解 通过 CSS 样式 打印 指定 内 容 的 
操作 过 程 : 首先 编写 用 于 控制 指定 内 容 的 不 打印 CSS 样式 。 程 序 的 代码 如 下 : 


<style> 
@media print{ 
div{display:none} 
.bgnoprint{ 
background:display:none: 
} 
.noprint{ 
display:none 
} 
} 
body { 
margin-top: Opx; 
} 
</style> 


然后 设计 页 面 的 布局 ， 将 页 面 的 头 文件 和 尾 文件 以 插入 背景 图 片 的 方式 进行 实现 ， 连 接 数 据 库 ， 
应 用 while 循环 语句 输出 数据 库 中 的 客户 信息 , 并 且 指 定 页 面 的 头 文件 和 尾 文件 为 不 打印 内 容 。 程序 的 
代码 如 下 : 


<body class="bgnoprint"> 
< 应 用 bgnoprint 样 式 一 一 > 
<table width="935" height="145" border="0" cellpaddine="0" cellspacine="0"> 
<t> 
<!-- 设置 头 文件 ， 以 背景 图 像 的 形式 进行 输出 ， 并 且 应 用 bgnoprint 样 式 ， 控 制 该 背景 图 像 为 不 打印 的 内 容 --> 
<td background="images/index_03.gif" class="benoprint">&nbsp:</td> 
</t> 
</table> 
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eI - -一 一 连接 数据 库 ， 循 环 输出 数据 库 中 数据 ~ - i 
<?php 
include("conn/conn.php"): // 连 接 数 据 库 
$query=mysql query("select * 位 om tb_customer "): /执行 查询 语句 
while(Smyrow=mysql fetch array(Squery){ // 循 环 输出 数据 库 中 的 内 容 
?> 
<t> 


<td height="26" align="center" bgcolor="#FFFFFF"><?php echo $myrow[customer user]:?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo Smyrow[customer tel]:?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo Smyrow[customer address]:?></td> 
<td align="center" bgcolor="#FFFFFF"><a href="delete_customer.php?delete=<?php echo 
Smyrow[customer_id]:?>"> 删 除 </a></td> 
</t> 
<?php }7> 
</table> 


最 后 设置 一 个 div 标签 ， 建 立 超 链接 ， 应 用 onclick 事件 调用 window.print0 方 法 和 printview0 函 数 
实现 页 面 的 打印 和 打印 预览 的 功能 。 
<!-- 设 置 超 链 接 ， 应 用 onclick 事 件 ， 调 用 window.print 方 法 执行 打印 ， 调 用 printview0 函 数 实现 打印 预览 的 功能 --> 
<t> 
<td height="26" colspan="4" align="center" bgcolor="#FFFFFF"><div><a href="#" onClick="window.print0:"> 客 
户 详 单 打印 </a>&nbsp:<a hre 伍 "#" onClick="printview0:"> 打 印 预览 </a></div></td> 
</t> 


通过 CSS 样式 实现 页 面 中 指定 内 容 打 印 的 运行 结果 如 图 6.40 和 图 6.41 所 示 。 


客户 姓名 电话 联系 地 址 操作 
ET 0431-12345678 长 春 市 遇 除 
李 +* 13604338784 四 平市 提交 
客户 详 单打 印 打印 预览 
佬 物 帝 配送 信息 网 版 权 所 有 


图 6.40 客户 信息 打印 页 面 的 运行 结果 


当 打 印 预览 


EE 


本 | 


1-12345678 
13604336784 


图 6.41 客户 信息 打印 预览 页 面 的 的 运行 结果 
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技巧 : 在 进行 Web 打印 时 ， 可 以 通过 以 下 操作 控制 是 否 打印 背景 颜色 和 图 像 。 在 下 窗口 中 ， 选 
择 “ 工 具 ”/“Internet 选项 ”命令 ， 在 弹出 的 “Internet 选项 ”对 话 框 中 选择 “高 级 ”选项 
卡 ， 在 “设置 ”列表 中 设置 “打印 背景 颜色 和 图 像 ” 前 面 的 复 选 框 是 否 选中 ， 如 果 选 中 ， 
代表 打印 背景 颜色 和 图 像 ， 否 则 不 打印 背景 颜色 和 图 像 。 


6.12 本 章 总 结 


本 章 从 某 物流 配送 公司 的 实际 需求 角度 出 发 ， 开 发 一 个 完整 的 物流 配送 信息 管理 系统 ， 详 细 地 讲 
解 了 物流 配送 信息 网 的 开发 流程 ， 从 最 初 的 需求 分 析 、 可 行 性 分 析 ， 到 系统 的 设计 、 数 据 库 的 设计 ， 
其 中 重点 突出 车 源 信息 查询 模块 、 发 货 单 管理 模块 和 回执 单 验收 管理 模块 的 设计 。 通 过 本 章 的 学 习 ， 
读者 不 但 可 以 了 解 物 流 配送 系统 开发 的 整体 思路 ， 而 且 能 够 掌握 很 多 关键 的 技术 和 技巧 ， 包 括 函数 的 
灵活 运用 ;MySQL 存储 过 程 的 创建 和 应 用 ; 报表 打印 技术 的 实现 ; 应 用 正则 表达 式 验证 电话 号 码 等 。 


第 章 


学 校 图 书馆 管理 系统 


(Apache+PHP+phpMyAdmin+MYySQL 5.0 实现 ) 
(器 + 视频 讲解 : 2 小 时 38 分 钟 ) 


随 着 网 络 技 术 的 高 速 发 展 和 计算 机 应 用 的 普及 , 利用 计算 机 对 图 书馆 的 日 常 工 
作 进 行 管理 势 在 必 行 。 虽然 目前 很 多 大 型 的 图 书馆 已 经 有 一 整套 比较 完善 的 管理 系 
统 ， 但 是 在 一 些 中 小 型 的 图 书馆 中 ， 大 部 分 工作 仍 需 由 手工 完成 ， 工 作 起 来 效率 比 
较 低 ， 管 理 员 不 能 及 时 了 解 图 书馆 内 各 类 图 书 的 借阅 情况 ， 读 者 需要 的 图 书 难以 在 
短 时 间 内 找到 ， 不 便于 动态 及 时 地 调整 图 书 结 构 。 为 了 更 好 地 适应 当前 读者 的 借阅 
需求 ,解决 手工 管理 中 存在 的 许多 现 端 ， 越 来 越 多 的 中 小 型 图 书馆 正在 逐步 向 计算 
机 信息 化 管理 转变 。 本 章 通 过 开发 一 个 流行 的 图 书馆 管理 系统 ， 为 读者 讲解 详细 的 
项 目 开发 流程 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


图 书馆 管理 系统 开发 的 基本 过 程 
系统 设计 的 方法 

如 何 分 析 并 设计 数据 库 、 数 据 表 
多 表 查 询 的 方法 
面向 对 象 的 编程 方法 

主要 功能 模块 的 实现 方法 

单元 测试 的 方法 

如 何 自动 计算 图 书 归 还 日 期 

内 联接 和 外 联接 语句 的 使 用 方法 


浊音 有 音 理 理 间 有理 总 达 
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7 汪 入 和 党 


博 考 图 书馆 是 吉林 x X 师范 学 校 的 大 型 图 书馆 。 随 着 学 校 图 书馆 规模 的 不 断 壮大 ， 经 营 的 图 书 品 
种 、 数 量 也 逐渐 增多 。 在 学 校 图 书馆 不 断 发 展 的 同时 ， 校 图 书馆 常年 采用 的 传统 的 人 工 方式 管理 暴露 
了 一 些 问题 。 例 如 ， 查 找 读者 借阅 的 某 一 本 图 书 的 具体 摆 放 位 置 ， 需 要 靠 人 工 记忆 在 书 海中 苦 苦 查 找 ， 
由 于 图 书 储存 量 大 ， 很 难 准确 定位 图 书 的 具体 位 置 ， 因 此 每 天 都 要 浪费 大 量 宝贵 的 时 间 资 源 。 学 校 图 
书馆 为 提高 工作 效率 ， 同 时 摆脱 图 书 管理 人 员 在 工作 中 出 现 的 种 种 整 端 ， 现 需要 委托 某 单位 开发 一 个 
学 校 图 书馆 管理 系统 。 


7.2 需求 分 析 


通过 计算 机 对 图 书 进行 管理 ， 不 仅 为 图 书馆 的 管理 注入 了 新 的 生机 ， 而 且 在 运营 过 程 中 节省 了 大 
量 的 人 力 、 物 力 、 财 力 和 时 间 ， 不 仅 提高 了 管理 员 的 效率 ， 还 为 图 书馆 在 读者 群 中 树立 了 一 个 全 新 的 
形象 ， 为 图 书馆 日 后 发 展 黄 定 一 个 良好 的 基础 。 通 过 对 一 些 大 型 图 书馆 的 实际 考察 、 分 析 ， 并 结合 
书馆 的 要 求 以 及 实际 的 市 场 调查 ， 要 求 本 系统 具有 以 下 功能 : 

网 站 设计 页 面 要 求 美观 大 方 、 个 性 化 ， 功 能 全 面 ， 操 作 简单 。 

要 求实 现 基础 信息 的 管理 平台 。 

要 求 对 所 有 读者 进行 管理 。 

要 求实 现 图 书 借阅 排行 、 了 解 当前 的 畅销 书 。 

商品 分 类 详尽 ， 可 按 不 同类 别 查 看 图 书信 息 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 ， 保 证 数据 查询 的 灵活 性 。 

实现 图 书 借阅 、 图 书 续 借 、 图 书 归 还 的 功能 。 

实现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查询 等 。 
要 求 图 书 借阅 、 续 借 、 归 还 时 记 下 每 一 笔记 录 的 操作 员 。 

实现 对 图 书 借阅 、 续 借 和 归还 过 程 的 全 程 数据 信息 跟踪 。 

提供 借阅 到 期 提醒 功能 ， 使 管理 者 可 以 及 时 了 解 到 已 经 到 达 归 还 日 期 的 图 书 借阅 信息 。 
提供 灵活 、 方 便 的 权限 设置 功能 ， 使 整个 系统 的 管理 分 工 明确 。 

具有 易 维 护 性 和 易 操作 性 。 


办 办 办 办 多多 欠 多 多 多 凶 凶 凶 


7.3 系统 设计 


7.3.1 系统 目标 


根据 前 面 所 作 的 需求 分 析 及 用 户 的 需求 可 以 得 出 ,学校 图 书馆 管理 系统 实施 后 ， 应 达到 以 下 目标 : 
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因 罗 办 愉 办 办 办 办 办 办 办 多 多 风 


7¥:322 


网 站 设计 页 面 要 求 美观 大 方 、 功 能 全 面 ， 操 作 简单 。 

网 站 整体 结构 和 操作 流程 合理 顺畅 ， 实 现 人 性 化 设计 。 
规范 、 完 善 的 基础 信息 设置 。 

对 操作 员 设置 不 同 的 操作 权限 ， 为 管理 员 提 供 修改 权限 功能 。 
对 所 有 读者 进行 集中 管理 。 

对 图 书信 息 进行 集中 管理 。 

实现 图 书 借阅 排行 ， 以 便 了 解 当前 的 畅销 书 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 。 

实现 图 书 借阅 、 图 书 续 借 、 图 书 归 还 功能 。 

实现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查 询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查 询 等 。 
实现 图 书 借阅 、 续 借 、 归 还 时 记 下 每 一 笔记 录 的 操作 。 

支持 图 书 到 期 提醒 功能 。 

为 操作 员 提 供 密码 修改 功能 。 

系统 运行 稳定 、 安 全 可 靠 。 


系统 功能 结构 


根据 学 校 图 书馆 管理 系统 的 特点 ， 可 以 将 其 分 为 系统 设置 、 读 者 管理 、 图 书 档案 管理 、 图 书 借 还 、 
系统 查询 等 5 个 部 分 ， 其 中 各 个 部 分 及 其 包括 的 具体 功能 模块 如 图 7.1 所 示 。 


7.3.3 


学 校 图 书馆 管理 系统 
图 书 档案 管理 


部 二 起 击 西 
瞳 奢 蛙 并 冰 下 
芋 阶 漆 萨 落 区 


疯 磅 漆 著 并 落 


图 7.1 学 校 图 书馆 管理 系统 功能 结构 图 


系统 流程 图 


学 校 图 书馆 管理 系统 的 流程 如 图 7.2 所 示 。 
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vy vy 
图 书 借阅 | | 图 书 续 借 | 图 书 归还 
[ | 


Ts 
| 


获取 读者 信息 pe 
[EE 


图 7.2 学 校 图 书馆 管理 系统 流程 图 
7.3.4 系统 预览 


学 校 图 书馆 管理 系统 由 多 个 程序 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 可 参见 光盘 中 的 
源 程序 。 

系统 登录 页 面 如 图 7.3 所 示 ， 该 页 面 用 于 实现 管理 员 登 录 。 系 统 首 页 如 图 7.4 所 示 ， 该 页 面 用 于 实 
现 显示 系统 导航 、 图 书 借阅 排行 和 版 权 信息 等 功能 。 


@ a 


© Ld 2 : 
NS SN ren Ba Wi ND 
wn Test aN re pe 


ran gg ree whet rm i 
Bibel nde loot) 


图 7.3 系统 登录 (光盘 \TM\07\library\login.php) 图 7.4 系统 首页 (光盘 \TM\07\library\index.php) 
图 书 借阅 页 面 如 图 7.5 所 示 ， 该 页 面 用 于 实现 图 书 借阅 功能 。 图 书 借阅 查询 页 面 如 图 7.6 所 示 ， 该 


> 
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页 面 用 于 实现 按照 复合 条 件 查询 图 书 借阅 信息 。 


@@ “和 3 到 4 位 全 


图 7.5 图 书 借阅 (光盘 \TM\07\ibrary\bookBorrow.php) 图 7.6 图 书 借阅 查询 (光盘 \TMWO7Nlibrary\borrowQuery.php) 
7.3.5 开发 环境 


在 开发 学 校 图 书馆 管理 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 
.服务 器 端 
操作 系统 : Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.2。 
PHP 软件 : PHP 5.1.6。 
数据 库 ， MySQL 5.0.24。 
MySQL 图 形 化 管理 软件 : phpMyAdmin-2.9.0.2。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 ， 最 佳 效 果 1024X768 像素 。 
客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 1024X768 像素 。 


7.3.6 文件 夹 组 织 结构 


a 


办 办 办 办 办 办 办 轨 


» 


加 加 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 〈 例 如， 创建 一 个 名 为 Images 的 文 
件 夹 ， 用 于 保存 网 站 中 所 使 用 的 图 片 )》 ， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 
架构 。 笔 者 在 开发 学 校 图 书馆 管理 系统 时 ， 设 计 了 如 图 7.7 所 示 的 文件 夹 组 织 结构 图 。 在 开发 时 ， 只 需 
要 将 所 创建 的 文件 保存 在 相应 的 文件 夹 中 就 可 以 了 。 
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日 转 litrsry 
日 全 ”com 一 用 于 存 信 网 站 使 用 的 数据 库 连 接 文件 
机 四 “= 一 用 于 存 信 网 站 使 用 的 CSS 样 式 表 
下 四 at 一 一 一 一 一 一 一 一 一 一 一 用 于 存 傅 网 站 使 用 的 数据 库 文 件 
相国 ass 一 一 一 一 一 一 一 一 一 一 用 于 存储 网 站 使 用 的 图 片 资源 文件 
日 情 下 一 一 一 一 一 一 一 一 一 一 一 用 于 存 针 网 站 使 用 的 JavsSeript 针 本 文件 


图 7.7 文件 夹 组 织 结构 
7.4 数据 库 设 计 


学 校 图 书馆 管理 系统 是 一 个 数据 库 开 发 的 Web 网 站 。 下 面 对 学 校 图 书馆 使 用 的 数据 库 进行 分 析 和 
介绍 。 


7.4.1 数据 库 分 析 


由 于 本 系统 是 为 中 小 型 的 图 书馆 开发 的 程序 ， 因 此 需要 充分 考虑 到 成 本 及 使 用 需求 〈 如 跨 平 台 ) 
等 问题 ， 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 的 跨 平台 的 关系 型 数据 库 
系统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


7.4.2 ”数据库 概念 设计 


根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 图 书 档 
案 实 体 、 读 者 档案 实体 、 借 阅 档 案 实体 、 归 还 档案 实体 和 管理 员 实 体 。 下 面 将 介绍 几 个 关键 实体 的 E-R 图 。 

1， 图书 档案 实体 

图 书 档案 实体 包括 编号 、 条 形 码 、 书 名 、 类 型 、 作 者 、 译 者 、 出 版 社 、 价 格 、 页 码 、 书 架 、 录 入 
时 间 和 操作 员 等 属性 。 图 书 档案 实体 的 E-R 图 如 图 7.8 所 示 。 


7.8 图 书 档案 实体 E-R 
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2. 读者 档案 实体 
读者 档案 实体 包括 编号 、 姓 名 、 人 性别、 条形码、 职业、 出 生日 期 、 有 效 证 件 、 证 件 号 码 、 电 话 、 
电子 邮件 、 登 记 日 期 、 操 作 员 、 类 型 和 备注 等 属性 。 读 者 档案 实体 的 E-R 图 如 图 7.9 所 示 。 


电子 邮件 


图 7.9 读者 档案 实体 E-R 图 
3. 借阅 档案 实体 
借阅 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 借 书 时 间 、 应 还 时 间 、 操 作 员 和 是 否 归 还 等 属性 。 
借阅 档案 实体 的 E-R 图 如 图 7.10 所 示 。 
4. 归还 档案 实体 
归还 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 归 还 时 间 和 操作 员 等 属性 。 归 还 档案 实体 的 E-R 
图 如 图 7.11 所 示 。 


归还 时 间 


图 7.10 借阅 档案 实体 E-R 图 图 7.11 “归还 档案 实体 E-R 图 
7.4.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需 求 的 分 析 , 学 校 图 书馆 管理 系统 db library 数据 库 主要 包含 如 下 9 个 数据 
表 ， 如 图 7.12 所 示 。 

结合 该 数据 表 的 创建 方法 ， 读 者 可 以 自行 创建 以 下 数据 表 。 数 据 表 的 设计 结构 如 图 7.13 一 图 7.15 
所 示 。 
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加 服务 器 : localhost ， 昌 数据 库 : db_library 

表 类 型 整理 说 明 
也 _bookcase 。 MyISAM ”gb2312_chinese_ci 。 图 书 书架 信息 表 
也 _bookinfo 。 MyISAM 。 gb2312_chinese_ci 。 图 书信 息 表 
也 _borrow MyISAM ”gb2312_chinese_ci 。 图 书 借阅 信息 表 
也 _library MyISAM ”gb2312_chinese_cl 。 图 书 过 信息 表 
也 _manager 。 MySAM 。 gb2312_chinese_ci 。 管理 员 信息 表 
也 _parameter ”MyISAM 。 gb2312_chinese_ci 。 参数 设置 信息 表 
也 _publishing 。 MySAM 。 gb2312_chinese_cl 。 出 版 社 信息 表 
也 _purview MISAM gb2312_chinese_ci 。 祝 限 信息 表 
也 _reader MySAM ”gb2312_chinese_ci 。 读者 信息 表 
tb_readertype MySAM ”gb2312_chinese_cl 读者 类 型 信息 表 


图 7.12 学 校 图 书馆 管理 系统 数据 表 
1. tb_bookinfo (图 书信 息 表 ) 
图 书信 息 表 主要 用 于 存储 图 书 的 基础 信息 。 该 数据 表 的 结构 如 图 7.13 所 示 。 


四 服务 器 : localhost 》 恒 数据 库 : db_library ， 国 表 : tb_bookinfo 
字段 大 型 整理 Null 额外 

barcode varchar(30) gb2312_chinese_ci 否 图 书 条 形 码 
bookname varchar(70) gb2312_chinese_cl 是 NULL 图 书 名 称 
‘typeid int(10) 是 NULL 图 书 类 型 
author varchar(30) gb2312_chinese_cl 是 WULL 图 书 作者 
transtator -= varchar(30) gb2312_chinese_cl 是 NULL 图 书 泽 者 
varchar(20) gb2312_chinese_ci 是 WULL 图 有 ISBN 
price Mloat(8,2) 是 NULL 图 书 定价 
page int(10) 是 NULL 图 书页 码 
bookcase int(10) 是 NULL 图 书 书架 
storage int(10) 是 NULL 图 书库 存 
inTime date 是 WNULL 入 库 时 间 
operator 。 varchar(30) gb2312_chinese_ci 是 NULL 操作 员 
del tinyint(1) 是 NULL 是 否 删除 
地 int(11) 是 。 NULL auto_increment 自动 编 Sid 


图 7.13 图 书信 息 表 结 构 
2. tb_borrow〔 图 书 借阅 信息 表 ) 
图 书 借阅 信息 表 主 要 用 于 存储 图 书 的 借阅 信息 。 该 数据 表 的 结构 如 图 7.14 所 示 。 


胃 服务 器 : localhost ， 避 数据 库 : db_library 、 国 表 : tb_borrow 

字段 类 型 整理 Null 默认 格外 说 明 
a int(10) 理 auto_increment 。 自动 编号 这 
readerid int10) 是 WNULL 读者 ad 哺 号 
bookid int10) 是 NULL 图 书 :编号 
borrowTime date 是 NULL 图 书 异 阅 时 间 
backTime date 是 WULL 图 书 归还 时 间 
‘operator varchar(30) gb2312_chinese_ci 是 NULL 操作 员 
imback tinyint(t) 是 0 是 否 归 还 


图 7.14 图 书 借阅 信息 表 结 构 
3. tb_reader (读者 信息 表 ) 
读者 信息 表 主 要 用 于 存储 用 户 的 读者 的 基础 信息 。 该 数据 表 的 结构 如 图 7.15 所 示 。 


第 7 章 学 校 图 书馆 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


图 服务 器 : localhost ， 昌 db_library ， 国 表 :tb_reader 

字 肪 类 型 Null 著 认 额外 说 明 
到 int(10) 否 auto_increment ”自动 编号 这 
name varchar(20) 。 gb2312_chinese cl 是 NULL 读者 姓名 
Sex varchar(4) gb2312_chinese_ci 是 NULL 性 别 
barcode varchar(30) 。 gb2312_chinese_cl 是 NULL 读者 条 形 码 
vocation varchar(50) 。 gb2312_chinese_ci 是 WULL 读者 职业 
birthday date 是 NULL 出 生日 其 
paperType -varchar(10) gb2312_chinese_cl 是 NULL 读者 证 件 
paperNO varchar(20) 。 gb2312_chinese_ci 是 NULL 证 件 号 码 
tel varchar(20) 。 gb2312_chinese_ci 是 NULL 读者 电话 
email varchar(100) gb2312_chinese_ci 是 NULL Email 也 址 
createDate date 是 NULL 办 卡 时 间 
‘operator varchar(30) 。 gb2312_chinese_ci 是 NULL 操作 员 
remark mediumtext 。 gb2312_chinese_cl 是 NULL 备注 
ypeid intdtT) 是 NULL 读者 类 型 


图 7.15 读者 信息 表 结 构 
[ 说明: 限于 篇 幅 ， 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 可 参见 本 书 附带 的 光盘 。 


7.5 首页 设计 


7.5.1 首页 概述 


管理 员 通 过 “系统 登录 ”模块 的 验证 后 ， 可 以 登录 到 图 书馆 管理 系统 的 首页 。 系 统 首页 主要 包括 
导航 栏 、 排 行 榜 和 版 权 信息 3 部 分 。 其 中 ， 导 航 栏 中 的 功能 菜单 将 根据 登录 管理 员 的 权限 进行 显示 。 
例如 ， 系 统管 理 员 Tsoft 登录 后 ， 将 拥有 整个 系统 的 全 部 功能 ， 因 为 它 是 超级 管理 员 。 

下 面 看 一 下 本 案例 中 提供 的 系统 首页 , 该 页 面 在 本 书 光盘 中 的 路 径 为 光盘 \TM\07\library\index.php， 
如 图 7.16 所 示 。 


忆 理 系 
(sds th re 


= 打下 号 三 借阅 排行 榜 
排名 图书 条 形 码 图 书 名 称 [ 出 版 入 

1 。 957654321 Pi 函数 参考 大 全 至 奥 系列。 PE 书架。 人 民 闻 电 出 版 村 ”部 和 、 张 m 
2 123456799 PIP 数据 库 系 统 开发 充 全 手册 数据库 技 术 。 DIE 书架 ”人 | 出 版 社 证 刘 ” 
3 123454321 Pg 程序 开发 否 例 三 和 下 。 ”三 开 列 。 Pr 加 | 排行 榜 
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本 站 请 使 用 IF 6.0 或 以 上 版 本 1024+T85 为 景 佳 旦 示 效 果 


图 7.16 学 校 图 书馆 管理 系统 首页 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


7.5.2 首页 技术 分 析 


学 校 图 书馆 管理 系统 是 一 个 功能 全 面 、 大 型 的 Web 网 站 ， 通 过 对 网 站 的 安全 性 考虑 ， 本 网 站 对 该 
系统 进行 权限 的 分 配 ， 只 有 管理 员 级 别 的 超级 用 户 可 以 对 普通 用 户 的 权限 进行 管理 和 设置 。 系 统 首 页 
主要 通过 判断 管理 员 的 权限 来 显示 该 用 户 所 操作 的 功能 模块 ， 关 键 代 码 如 下 : 

例 程 01 代码 位 置 : 光盘 \TM\07\library\navigation.php 


<?php 
Session start(): /初始 化 session 变 量 
include("conn/conn.php"):; // 连 接 数 据 库 文件 


$query=mysql_query("select m.id,m.name.p.id.p.sysset,p.readerset.p.bookset.p.borrowback,p.sysquery from tb _ manager 
as Im left join (select * from tb_purview ) as p on m.id=p.id where name='$_SESSION[admin name]"); 

Sinfo=mysql_fetch_array($query): // 检 索 用 户 权限 

?> 

<!-- 检 索 用 户 所 对 应 的 权限 ， 如 果 权限 值 为 1， 则 说 明 该 功能 可 用 ， 并 输出 到 浏览 器 ， 否 则 不 显示 --> 

<td width="70%" align="right"> 

<a href="index.php" class="al"> 首 页 </a> | 

<?php if($info[sysset]==D){ ?><a onmouseover=showmenu(event,sysmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="al"> 系 统 设置 </a> | <?php}?> 

<?php if(Sinfo[readerset]=——1){?><a ‘onmouseover=showmenu(event,readermenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="al"> 读 者 管理 </a> | <?php}?> 

<?php if(Sinfo[bookset]==1){ ?><a href="book.php" class="al"> 图 书 档案 管理 </a> | <?php }?> 

<?php if(Sinfo[borrowback]—1){?><a onmouseover=showmenu(event,borrowmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand"class="al" > 图 书 借 还 </a> | <?php}?> 

<?php ifSinfo[sysquery] 一 D{ ?><a onmouseover=showmenu(event,querymenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="al"> 系 统 查询 </a> | <?php}?> 

<a ”href="pwd_Modify.php" class="al"> 更 改口 令 </a> | 

<a href="safequit.php" class="al"> 注 销 </a> 

</td> 


[四 说 明 : 在 权限 信息 表 tb_purview 中 ， 权 限 值 为 1， 代 表 具备 该 模块 的 操作 权限 ; 权限 值 为 0， 代 表 
不 具备 该 模块 的 操作 权限 。 
在 实现 系统 导航 菜单 时 ， 引 用 了 JavaScript 文件 menu.JS， 该 文件 中 包含 全 部 实现 半 透 明 背 景 菜单 
的 JavaScript 代码 。 


避 技巧 : 将 页 面 中 所 涉及 的 JavaScript 代码 保存 在 一 个 单独 的 JS 文件 中 ， 然 后 通过 <script></script> 将 
其 引用 到 需要 的 页 面 ， 可 以 规范 页 面 代 码 。 在 系统 导航 页 面 引 用 menu.JS 文件 的 代码 如 下 : 


<script src="JS/menu.JS"></script> 
7.5.3 首页 的 实现 过 程 


系统 首页 的 内 容 显示 区 用 于 显示 图 书 的 排行 信息 ， 并 将 排行 结果 按 借阅 数量 降序 排列 。 该 页 的 关 
键 代 码 如 下 : 
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例 程 02 ”代码 位 置 :光盘 VTMWO7libraryindex php 

<?php 

include("conn/conn.php"): // 连 接 数 据 源 文件 

$sql=mysql query("select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr join 
(select b.*.c.name as bookcasename.p.pubname,tLtypename from tb_bookinfo b left join tb_bookcase c on b.bookcase=c.id join 
tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book on borr.bookid=book.id order 
by borr.degree desc limit 10"): 


S$info=mysql fetch array($sql): // 检 索 图 书 借阅 信 息 

$i=1; 

dof{ // 应 用 do…while 循 环 语句 显示 图 书信 息 
?> 

<t> 


<td height="25" align="center"><?php echo $i:?></td> 

<td style="padding:5px:">&nbsp;<?php echo Sinfo[barcode]:?></td> 
<td style="padding:5px:"><?php echo $info[bookname]:?></td> 

<td style= "padding:5px:"><?php echo $info[typename]:?></td> 

<td align="center">&nbsp:<?php echo $info[bookcasename]:?></td> 
<td align="center">&nbsp:;<?php echo $info[pubname];?></td> 

<td align="center"><?php echo $info[author]:?></td> 

<td align="center"><?php echo $info[price]:?></td> 

<td align="center"><?php echo $info[degreel:?></td> 


<t> 

<?php 
Si=$itl; // 变 量 自 加 1 操作 
}while(Sinfo=mysql_fetch_array(SsqD); /do…while 循 环 语句 结束 


7.6 管理 员 模 块 设计 


7.6.1 管理 员 模块 概述 


管理 员 模块 主要 包括 管理 员 登 录 、 查 看 管理 员 列表 、 添 加 管理 员 信息 、 管 理 员 权限 设置 、 管 理 员 
删除 和 更 改口 令 等 6 个 功能 。 管 理 员 模 块 的 框架 如 图 7.17 所 示 。 


7.6.2 ”管理 员 模块 技术 分 析 


在 管理 员 模 块 中 , 涉及 到 的 数据 表 是 tb_manager (管理 员 信息 表 ) 和 tb_purview (权限 表 ) 。 其 中 ， 
管理 员 信息 表 中 保存 的 是 管理 员 名 称 和 密码 等 信息 ， 权 限 表 中 保存 的 是 各 管理 员 的 权限 信息 ， 这 两 个 
表 通 过 各 自 的 id 字段 相关 联 。 通 过 这 两 个 表 可 以 获得 完整 的 管理 员 信息 。 

各 注意: 在 实现 系统 登录 前 ,需要 在 MySQL 数据 库 中 ， 手 动 添加 一 条 系统 管理 员 的 数据 〔 管 理 员 
名 为 Tsoft、 密 码 为 111、 拥 有 所 有 权限 ) ， 即 在 MySQL 的 客户 端 命令 行 中 应 用 下 面 的 语 
多 分 别 向 管理 员 信息 表 tb_manager 和 权限 表 tb_purview 中 各 添加 一 条 数据 。 
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S v 了 y 
添加 管理 员 权限 设置 | 删除 管理 员 | 更 改口 令 


图 7.17 管理 员 模 块 的 框架 图 


# 洪 加 管理 员 信息 

insert into tb_ manager (name,pwd) Values(CTsoff 1117): 
太 潍 加 权限 信息 

insert into tb_purview values(1.1.1.1.1.1): 


从 网 站 安全 的 角度 考虑 ， 仅 仅 有 上 面 介绍 的 系统 登录 页 面 并 不 能 有 效 地 保存 系统 的 安全 ， 一 旦 系 
统 首页 面 的 地 址 被 他 人 获得 ， 就 可 以 通过 在 地 址 栏 中 输入 系统 的 首页 面 地 址 而 直接 进入 到 系统 中 。 为 
了 便于 网 站 的 维护 ， 因 此 将 验证 用 户 是 否 登录 的 代码 封装 在 独立 的 PHP 文件 中 ， 即 check_login.php 文 
件 。 验 证 用 户 是 否 登录 的 具体 代码 如 下 : 


例 程 03 ”代码 位 置 ; 光盘 \TMWO7Nibraryvcheck login .php 


<?php 
session_start(): // 初 始 化 session 变 量 
这 $_ SESSION[admin name]—""){ /如果 session 变 量 为 空 ， 则 说 明 用 户 未 登录 


echo "<script>alert( 对 不 起 ， 请 通过 正确 的 途径 登录 博 考 图 书馆 管理 系 
统 !"):window.location.href='login.php':</script>":; 
?> 
当 系 统 调 用 首页 时 ， 会 判断 session 变量 admin_ name 是 否 存在 ， 如 果 不 存 在 ， 将 页 面 重 定 向 到 系 
统 登 录 (login.php) 页 面 。 


7.6.3 ”系统 登录 的 实现 过 程 

国 ”系统 登录 使 用 的 数据 表 : tb manager 

系统 登录 是 进入 学 校 图 书馆 管理 系统 的 入 口 ， 主 要 用 于 验证 管理 员 的 身份 。 运 行 本 系统 ， 首 先进 
入 的 是 系统 登录 页 面 ， 在 该 页 面 中 ， 系 统管 理 员 可 以 通过 输入 正确 的 管理 员 名 称 和 密码 登录 到 系统 首 
页 ， 当 用 户 没有 输入 管理 员 名 称 或 密码 时 ， 系 统 会 通过 JavaScript 进行 判断 ， 并 给 予 信息 提示 。 系 统 登 
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录 页 面 的 运行 结果 如 图 7.18 所 示 。 


@ 二 吉 周二 亿 管 理 系统 


BO KAO LIBRARY waa term 


图 7.18 系统 登录 页 面 的 运行 结果 


系统 登录 页 面 主要 用 于 收集 管理 员 的 输入 信息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 否 
为 空 。 该 页 面 中 亡 涉 及 的 表单 元 素 如 表 7.1 所 示 。 


表 7.1 系统 登录 页 面 所 涉及 的 表单 元 素 


名 称 含义 
forml 管理 员 登 录 表单 
name 管理 员 名 称 
pwd 管理 员 密 码 
submit | submit | Ee onclick="return check(form1)" “确定 ”按钮 
sbmi3 | rset | vane- 重 村 | “ 生 置 " 技 负 
submit2 | buton | Value=" 二 onClick= "window.closeO:” “关闭 ”按钮 


编写 自 定义 的 JavaScript 函数 ， 用 于 判断 管理 员 的 名 称 和 密码 是 否 为 空 ， 代 码 如 下 : 


例 程 04 ”代码 位 置 : 光盘 \VTMW7Nibrarylogin.php 
<script language="javascript"> 
function check(form){ // 自 定义 一 个 JavaScript 函 数 check0 
if (form.name.value—""){ // 如 果 管 理 员 名 称 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 管理 员 名 称 !"):form.name .focusO:retum false; 


: 
if (form.pwd.value—""){ // 如 果 管 理 员 密 码 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 


alert(" 请 输入 密码 !"):form.pwd .focusO:return false: 
> 
} 


</script> 

提交 表单 到 数据 处 理 页 ， 页 面 中 为 了 防止 非法 用 户 进入 学 校 图 书馆 管理 系统 首页 ， 通 过 调用 类 
chkinput0 方 法 实现 判断 用 户 名 和 密码 是 否 正确 。 如 果 为 合法 用 户 ， 则 可 以 登录 学 校 图 书馆 管理 系统 的 
首页 ; 否则 ， 弹 出 相应 的 错误 提示 。 关 键 代码 如 下 : 
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例 程 05 ”代码 位 置 , 光盘 \TMNO7librarychklogin php 


<?php 
session start(): // 初 始 化 session 变 量 
$A name=$ POST[namel: // 接 收 表单 提交 的 用 户 名 
$A_pwd=$ POST[pwd]: // 接 收 表单 提交 的 密码 
@ casschkinput{ // 定 义 类 
Var $name; 
Var $pwd: 
function chkinput(Sx.$y){ /定义 一 个 方法 
Sthis->name=$x: /将 管理 员 名 称 传 给 类 对 象 Sthis->name 
Sthis->pwd=$y: // 将 管理 员 密 码 传 给 类 对 象 $this->pwd 
1 
function checkinputO{ 
include("conn/conn.php"): // 连 接 数 据 库 文件 
$sql=mysql_query("select * 位 om tb_manager Where name=".$this->name." and pwd=".$this->pwd."",$conn); 
Sinfo=mysql fetch_array($sq]): // 检 索 管理 员 名 称 和 密码 是 否 正确 
if($info=—false){ // 如 果 管 理 员 名 称 或 密码 不 正确 ， 则 弹出 相关 提示 信息 


echo "<script language='javascript>alert( 您 输入 的 管理 员 名称 错 误 ， 请 重新 输入 ! 
:history.back();</script>"; 
exit; 
} 
else{ // 如 果 管理 员 名 称 或 密码 正确 ， 则 弹出 相关 提示 信息 
echo "<script>alert( 管 理 员 登录 成 功 !"):window.location=index.php';</script>"; 
$_SESSION[admin_name]=S$info[name]: 。“// 将 管理 员 名 称 存 到 $_SESSION[admin_name] 变 量 中 


$_SESSION[pwd]=$info[pwd]: // 将 管理 员 密 码 存 到 $_SESSION[pwd] 变 量 中 
} 
} 
} 
@ S$obj=new chkinput(trim($name),trim($pwd)): // 创 建 对 象 
@ S$obj->checkinput0; /调用 类 


?> 


Ah 代码 贴 十 

@ class: 创建 一 个 PHP 类 时 ， 必 须 使 用 关键 字 class 进行 声明 ， 该 关键 字 后 紧 跟 类 的 名 称 ， 之 后 用 大 括号 将 类 体 进 
行 封装 。 

@ new: PHP 中 应 用 new 关键 字 创 建 对 象 。 该 模块 创建 了 一 个 名 为 chkinput 的 验证 管理 员 类 。 

@ $obj->checkinput(0:: 通过 $obj 对 象 调用 checkinput 类 中 的 属性 和 方法 ， 即 管理 员 名 称 和 密码 。 


7.6.4 查看 管理 员 的 实现 过 程 


国 ”查看 管理 员 使 用 的 数据 表 : tb manager、tb purview 

管理 员 登 录 后 ， 选 择 “ 系 统 设置 ”/“ 管 理 员 设置 ”菜单 项 ， 进 入 到 查看 管理 员 列 表 页 面 。 在 该 页 
面 中 ， 将 以 表格 的 形式 显示 全 部 管理 员 及 其 权限 信息 ， 并 提供 添加 管理 员 信息 、 删 除 管理 员 信息 和 设 
置 管理 员 权 限 的 超 链接 。 查 看 管理 员 列 表 页 面 的 运行 结果 如 图 7.19 所 示 。 
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情 ) 拍 亏 图 媳 馆 管理 系统 
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图 7.19 查看 管理 员 列表 页 面 的 运行 结果 


首先 使 用 左 外 联接 语句 〈left join…on) 从 数据 表 tb_manager 和 tb_purview 中 查询 出 符合 条 件 的 数 
据 ， 然 后 将 查询 结果 应 用 do…while 循环 语句 输出 到 浏览 器 。 关 键 代码 如 下 : 


例 程 06 ”代码 位 置 ， 光盘 \TM\02\library\manager.php 
<?php 
include("conn/conn.php"):; // 连 接 数 据 库 文件 
© $sql=mysql query("selectm.id.m.name.p.sysset.p.readerset.p.bookset,p.borrowback.p.sysquery from tb_manager as m 
left join (select * from tb_purview) as p on m.id=p.id"); 
Sinfo=mysql_fetch_array($sql): /检索 数据 信息 
do{ /应 用 do…while 循 环 语句 输出 查询 结果 
?> 
Vid 中 事由 中 中 中 让 中 中 字 刘 审 训 中 中 中 下 相让 事 本 本 下 可可 让 中 站 本本 本 革 人 于 符合 查询 条 件 的 记录 闻 相 机 中 相机 机 相机 本 机 机 可 本 事 训 机 本 事 下 本 下 本 可 可 可 事 事 可 束 事 束 */ 
<h> 
<td style="padding:5px:"><?php echo Sinfo[name];?></td> 
© <td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 
disabled="disabled" <?php if(S$info[sysset]—1){echo ("checked");}?>></td> 
<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 
disabled="disabled" <?php if(Sinfo[readerset]—1){echo("checked");}?>></td> 
<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled <?php 
if(Sinfo[booksetf]==1){echo("checked");}?>></td> 
<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled <?php 
if(Sinfo[borrowback]=——1){echo("checked");}?>></td> 
<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled <?php 
if(Sinfo[sysquery]—1){echo("checked");}?>></td> 
<td align="center"><a href="#" onClick="window.open(‘manager_ modify.php?id=<?php echo 
$info[id]: ?>",",'width=292.height=175")"> 权 限 设置 </a></td> 
<td align="center"><a href="manager_del.php?id=<?php echo $info[id]:?>"> 删 除 </a></td> 


</t> 
/本 洒 束 束 家 吾 事 于 守 束 于 家 于 于 束 守 字 束 环 守 于 守 字 训 环球 家 家 本末 于 于 字 家 训 家 于 于 于 字 家 末末 家/ 
<2?php 

}while(Sinfo=mysql fetch_array(SsqD); /do…while 循 环 语句 结束 
es 
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外 代码 贴 二 
@ leftjoin…on: 应 用 左 外 联接 将 两 个 表 或 多 个 表 连 接 起 来 ， 返 回 部 分 或 全 部 匹配 行 ， 详 解 参 见 7.10.2 节 相 关内 容 。 
@ <?php if($info[sysset] 一 1){echo ("checked"):}?>: 如 果 系 统 设置 字段 的 值 为 1， 则 复 选 框 处 于 选中 状态 。 


7.6.5 添加 管理 员 的 实现 过 程 


国 。 添加 管理 员 使 用 的 数据 表 : tb_ manager 

管理 员 登 录 后 ,选择 “系统 设置 ”/“ 管 理 员 设置 ”菜单 项 ， 入 http//127.0.0.1 -天 加 管理 负 信 息 加 秽 同 全 7 
进入 到 查看 管理 员 列 表 页 面 , 在 该 页 面 中 单 击 “添加 管理 信息 ” 
超 链接 ， 打 开 添加 管理 员 信息 页 面 。 添 加 管理 员 信息 页 面 的 运 
行 结果 如 图 7.20 所 示 。 
全 注意 : 新 添加 的 管理 员 信息 没有 权限 ， 必 须 通过 设置 管理 

员 权 限 为 其 指定 可 操作 的 功能 模块 。 

在 查看 管理 员 列 表 页 面 ， 单 击 “ 添 加 管理 信息 ” 超 链接 ， ”图 720 添加 管理 号 而 的 这 结果 
文字 的 HTML 代码 如 下 : 

例 程 07 ”代码 位 置 ， 光盘 \TM\02\library\manager.php 

<a href="#" onClick="window.open("manager_add.php'.",'width=292,height=175")"> 添 加 管理 员 信息 </a> 

添加 管理 员 页 面 主要 用 于 收集 输入 的 管理 员 信息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 合 法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 7.2 所 示 。 

表 7.2 添加 管理 员 页 面 所 涉及 的 表单 元 素 


管理 员 名 称 :图书 爱好 看 


fo 


submit value=" 保 存 " onClick="check(form1)" “保存 ”按钮 
button Value=" 关 闭 " onClick="window.closeO:" “关闭 ”按钮 


在 添加 管理 员 页 面 中 ， 输 入 合法 的 管理 员 名 称 及 密码 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数 
据 处 理 页 ， 将 添加 的 管理 员 信息 保存 到 数据 表 中 。 如 果 添 加 成 功 ， 弹 出 成 功 的 提示 信息 ; 否则 ， 弹 出 
错误 提示 ， 代 码 如 下 : 


例 程 08 ”代码 位 置 : 光盘 \TM\02\library\manager_ok.php 


<?php 

include("conn/conn.php"): // 连 接 数据 库 文件 

if($_POST[submit]!=""){ // 如 果 单 击 了 “保存 ”按钮 ， 则 执行 下 面 的 操作 
S$name=$_ POST[name]: /获取 管理 员 名 称 

Spwd=$_POST[pwd]: // 获 取 管 理 员 密 码 
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$sql=mysql_query("insert into tb_manager (name.pwd) values('$name'.'$pwd")"): 


if($sql——true){ /向 数据 表 中 添加 管理 员 信息 成 功 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert( 管 理 员 添加 成 功 ! ):window.close0:window.openerlocation .reload0:</script>"; 
9 

else{ // 向 数据 表 中 添加 管理 员 信息 失败 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添加 失败 ! ):window.close0:window.opener.location .reload():</script>"; 
} 

} 

?> 


ba 技巧 : 在 添加 管理 员 处 理 页 中 应 用 window.opener.location.reload(): 语 句 刷新 父 窗口 中 的 信息 


7.6.6 设置 管理 员 权 限 的 实现 过 程 http:/ /127.0.0.1/mr/schoolooon en [el Ed| 
| 省 理 员 名 称 图 和 要 好 痢 ] | 
国 ”设置 管理 员 权限 使 用 的 数据 表 : tb_manager、tb_purview i 证 
在 查看 管理 员 列 表 页 面 单 击 指定 管理 员 后 面 的 “权限 设置 ” aopmt 攻 te 
超 链接 ， 即 可 进入 到 “权限 设置 ” 页面， 设置 该 管理 员 的 操作 | El 
权限 。 权 限 设置 页 面 的 运行 结果 如 图 7.21 所 示 。 
权限 设置 页 面 中 所 涉及 到 的 表单 元 素 如 表 7.3 所 示 。 7.21 权限 设置 页 面 的 运行 结果 


表 7.3 权限 设置 页 面 所 涉及 的 表单 元 素 


Teaderset 
checkbox 


borrowback checkbox 2? 
sysquery checkbox 本 if($info[sysquery]==1) {echo("checked" yj 系统 查询 
submit submit class="btn_ grey"” “保存 ”按钮 


button Value=" 关 闭 " onClick="window.close0:" “关闭 ”按钮 


Submit2 


在 查看 管理 员 列 表 页 面 中 ， 添 加 “权限 设置 ” 列 ， 并 在 该 列 中 添加 以 下 用 于 打开 “权限 设置 ”页 
面 的 超 链 接 代码 。 

例 程 09 ”代码 位 置 : 光盘 \TM\02\library\manager.php 

<a href="#" onClick="window.open(‘manager_modify.php?id=<?php echo $info[id]: ?>"."."width=292.height=175)"> 权 限 
设置 </a> 

从 上 面 的 URL 地 址 中 可 以 获取 设置 管理 员 权 限 页 所 涉及 到 的 id 号 ,将 id 号 提交 给 处 理 页 
manager_modifyok.php， 修 改 id 号 所 对 应 的 管理 员 信息 。 具 体 代 码 如 下 : 
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例 程 10 代码 位 置 光盘 \TM\02\library\manager_modifyok.php 


<2php 
include("conn/conn.php"): // 连 接 数 据 库 文件 
if($_POST[submit]!=""){ // 如 果 提 交 表单 ， 则 执行 以 下 操作 
$id=$_POST[id]: // 获 取 id 信 息 
@ S$sysset=$ POST[sysset]—""?0:1; // 应 用 三 目 运 算 符 求 出 “系统 设置 ” 复 选 框 的 值 


Sreaderset=$ POST[readerset]—""?0:1; // 应 用 三 目 运算 符 求 出 “读者 管理 ” 复 选 框 的 值 


Sbookset=$ POST[bookset]—""?0:1; // 应 用 三 目 运算 符 求 出 “图 书 管理 ” 复 选 框 的 值 

Sborrowback=$ POST[borrowback]—""?0:1; // 应 用 三 目 运 算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 

Ssysquery=$ POST[sysquery] 一 ""20:1; /应 用 三 目 运算 符 求 出 “系统 查询 ” 复 选 框 的 值 

$query=mysql query("select * from tb_purview where id=$id"); 

Sinfo=mysql_fetch_array($query): // 检 索 权限 信息 表 中 是 否 存在 该 管理 员 

这 Sinfo 一 false){ /如 果 不 存在 ， 向 权限 表 中 添加 管理 员 权限 信息 
© mysql_query("insert into tb_purview(id,sysset.readerset,bookset.borrowback,sysquery) 


values($id, $sysset, Sreaderset, $bookset,$borrowback, $sysquery)"):; 


} 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 


mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery where id="$id"); 


echo"<script language=javascript>alert( 权 限 设置 修改 成 功 ! "):window.close(:window.opener.location.reload(:</script>"; 
// 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 
} 


?> 


亏 们 代码 贴 二 
@ $_ POST[sysset] 一 ""?0:1;: 应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 ， 如 果 等 于 空 ， 值 为 0， 否 则 值 为 1。 
@ insert…into: 向 权限 信息 表 中 添加 一 行 数据 信息 ，insert…into 语句 只 适用 于 对 单行 数据 的 插入 。 


7.6.7 ”删除 管理 员 的 实现 过 程 


国 。 删除 管理 员 使 用 的 数据 表 : tb_manager、tb_purview 

在 查看 管理 员 列 表 页 面 ， 单 击 指定 管理 员 信息 后 面 的 “删除 ” 超 链接 ， 该 管理 员 及 其 权限 信息 将 
被 删除 。 

在 查看 管理 员 列表 页 面 中 添加 以 下 用 于 删除 管理 员 信息 的 超 链接 代码 。 

例 程 11 代码 位 置 : 光盘 \TM\07\library\manager.php 

<a href="manager_del.php?id=<?php echo $info[id]:?>"> 删 除 </a> 

从 上 面 的 URL 地址 中 ,可 以 获取 删除 管理 员 所 涉及 到 的 id 号 ,将 id 号 提交 给 manager_del.php 处 
理 页 删除 id 号 所 对 应 的 管理 员 信 息 。 具 体 代 码 如 下 : 

例 程 12 代码 位 置 光盘 \TM\07\library\manager_del.php 


<?php 
include("conn/conn.php"): // 连 接 数 据 库 文件 
$id=$_GETI[id]: // 获 取 管 理 员 的 id 号 
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$sql-mysql query("delete from tb_ manager where id-'$id'"): /删除 管理 员 表 中 id 号 所 对 应 的 管理 员 信息 
$query=mysql_query("delete from tb_purview where id=$id"™"): // 删 除权 限 表 中 id 号 所 对 应 的 管理 员 权 限 
if($sql—true and Squery—true ){ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 


echo "<script language=javascript>alert( 管 理 员 删 除 成 功 ! "):history.back0:</script>"; 

} 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert( 管 理 员 删 除 失败 ! "):history.back0:</script>"; 

| 


> 


7.6.8 单元 测试 


在 开发 完 管理 员 模块 后 ， 对 该 模块 进行 单元 测试 。 当 管理 员 修 改 操作 员 的 权限 时 ， 该 操作 员 的 权 
限 没有 改变 。 下 面 给 出 修改 权限 设置 处 理 页 的 源 代码 : 


<2php 

include("conn/conn.php"): // 连 接 数 据 库 文件 
if($_POST[submit]!=""){ // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POSTI[id]: // 获 取 id 信 息 

Ssysset=$ POST[sysset]; // 获 取 “ 系 统 设置 ” 复 选 框 的 值 
Sreaderset=$_ POST[readerse{]; // 获 取 “ 读 者 管理 ” 复 选 框 的 值 
Sbookset=$_POST[bookset]; /获取 “ 图 书 管理 ” 复 选 框 的 值 
Sborrowback=$_ POST[borrowback]; // 获 取 “ 图 书 借 还 ” 复 选 框 的 值 
Ssysquery=S_POST[sysquery]; // 获 取 “ 系 统 查询 ” 复 选 框 的 值 
$query=mysql_query("select * from tb_purview where id=$id"); 

Sinfo=mysql_fetch_array($query): /检索 权限 信息 表 中 是 否 存在 该 管理 员 
这 $info 一 包 lse){ /如果 不 存在 ， 向 权限 表 中 添加 管理 员 权限 信息 


Imysql_query("insert into tb_purview(id,sysset,readerset,bookset,.borrowback,sysquery) 
values($id,$sysset,Sreaderset,$bookset.$Sborrowback.S$sysquery)"): 


} 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 
mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery ”where id="$id"); 

有 

echo"<script language=javascript>alert( 权 限 设置 修改 成 功 ! ):window.close0:window.openerlocation TeloadO:</script>": 

// 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 
} 


?> 

为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo0 语 句 对 $sysset 等 一 系列 传递 的 值 进行 输出 ， 输 出 的 结果 显 
示 $sysset 的 值 始终 等 于 0， 这 充分 说 明 在 修改 复 选 框 时 ， 没 有 获取 到 选中 复 选 框 的 值 。 

解决 该 问题 的 方法 对 提交 的 复 选 框 的 状态 进行 判断 ， 如果 复 选 框 处 于 未 选中 状态 ， 则 值 为 0; 如果 
复 选 框 处 于 选中 状态 ， 则 值 为 1。 这 样 就 可 以 获取 到 复 选 框 选中 状态 的 值 了 。 

处 理 复 选 框 的 状态 有 两 种 方法 : 一 种 是 应 用 让 条 件 语句 对 逐个 值 进行 判断 ， 判 断 复 选 框 的 当前 状 
态 ， 并 赋予 其 值 ( 但 这 种 方法 过 繁琐 ) ; 另 一 种 是 应 用 三 目 运算 符 ， 简 单 快捷 地 计算 复 选 框 的 值 这 
种 方法 可 以 简化 代码 ， 本 模块 采用 该 方法 获取 复 选 框 的 值 ) 。 

应 用 三 目 运 算 符 获 取 复 选 框 的 值 的 代码 如 下 : 
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Ssysset=$ POST[sysset]—""?0:1; // 应 用 三 目 运 算 符 求 出 “系统 设置 ” 复 选 框 的 值 
Sreaderset=$ POST[readerset]—""?0:1; 运算 符 求 出 “读者 管理 ” 复 选 框 的 值 
Sbookset=$ POST[bookset]—""?0:1; // 应 用 三 目 运 算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
Sborrowback=$ POST[borrowback]—""?0:1; 三 目 运 算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
Ssysquery=$ POSTI[sysquery]—""?0:1; // 应 用 三 目 运算 符 求 出 “系统 查询 ” 复 选 框 的 值 


完整 的 权限 设置 处 理 页 的 代码 参见 例 程 10。 
7.7 图 书 档案 管理 模块 设计 


7.7.1 图 书 档案 管理 模块 概述 


图 书 档案 管理 模块 主要 包括 查看 图 书 列表 、 添 加 图 书信 息 、 修 改 图 书信 息 、 删 除 图 书信 息 和 查看 
图 书 详细 信息 等 5 个 功能 。 图 书 档案 模块 的 框架 如 图 7.22 所 示 。 


EECIOTIIOTEIIII 


图 7.22 图书 档案 模块 的 框架 图 
7.7.2 图 书 档案 管理 模块 技术 分 析 
在 图 书 档案 管理 模块 中 ， 涉 及 到 的 数据 表 包 括 tb_bookinfo (图书 信息 表 ) 、tb_bookcase (书架 设 


置 表 ) 、tb booktype (图 书 类 型 表 ) 和 tb_publishing (出 版 社 信息 表 ) ， 这 4 个 数据 表 间 通过 相应 的 字 
段 进行 关联 ， 如 图 7.23 所 示 ， 通 过 以 上 4 个 表 可 以 获得 完整 的 图 书 档案 信息 。 


tb_booktype 


ia integer unsigned pk> | | i 
typenane varchar (30) id integer (ll1) Cpk> 
全 和 下 | bareode int 
booknane varchar (70) 
FRbooktype——ltypeid integer unsigned Cfk3> 
suthor varchar (30) 
tb_publishing FEK_pab translator varchar (30) 
varchar 0 ee- ISBN varchar (20) 2> 
Pabnane varchar (30) nies at, 2 
Page integer unsigned 
lbookcase integer unsigned <fkl> 
FRbookease | inTine date 
th_bookease operator varchar (30) 
id integer unsigned <pk> dal tinyint (1) 


name varchar 


图 7.23 图 书 档案 管理 模块 各 表 间 关系 图 
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公 注意 : 本 模块 主要 应 用 联接 语句 实现 多 表 查 询 ， 关 于 联接 语句 的 详细 讲解 参见 7.10 节 相 关内 容 。 
7.7.3 查看 图 书信 息 列表 的 实现 过 程 


国 ”查看 图 书信 息 列表 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 后 ， 选 择 “ 图 书 管理 ”/“ 图 书 档案 管理 ”菜单 项 ， 进 入 到 查看 图 书 列表 页 面 ， 在 该 页 
面 中 将 显示 全 部 图 书信 息 列 表 ， 同 时 提供 添加 图 书信 息 、 删 除 图 书信 息 、 修 改 图 书信 息 的 超 链 接 。 查 
看 图 书信 息 列表 页 面 的 运行 结果 如 图 7.24 所 示 。 


入】 停 记 周记 馆 管理 系统 


HO KAO TTptiaRY wawaxos ovavem 


当前 登录 的 用 户 ; Tsoft 


my 


尖 加 用 书信 息 
让 EX 扣除 

ng 煞 征 库 系 统 开 发 元 全 王 琐 Wn 
FE 程序 开 当 项 全 宇 身 Wy a 
re 卫 册 兆 才 大 全 we 


CopyRight 四 2007 we nrbeed com 吉林 yz? 师 范 大 学 图 书馆 
本 站 请 使 用 IE 6 0 或 以 上 版 本 1024*T66 为 最 佳 显示 效果 
图 7.24 查看 图 书信 息 列 表 的 运行 结果 
打开 功能 导航 navigation.php 文件 ， 设 置 “图 书 档案 管理 ”菜单 项 的 超 链接 的 代码 如 下 : 
例 程 13 ”代码 位 置 : 光盘 \TMVW2\librarymavigation.php 
<a hre 人 "book .php" class="al"> 图 书 档案 管理 </a> 
首先 应 用 join…on 内 联接 语句 将 tb_bookinfo、tb_bookcase、tb_booktype 和 tb_publishing 4 个 数据 
表 连 接 起 来 检索 指定 条 件 的 图 书信 息 , 然后 应 用 do…while 循环 语句 输出 查询 结果 到 浏览 器 。 查看 图 书 
信息 页 面 的 代码 如 下 : 


例 程 14 ”代码 位 置 : 光盘 \TM\07\library\book.php 


<?php 

include("conn/conn.php"): // 连 接 数 据 库 文 件 

$query=mysql_query("select book.barcode.bookid as bookid,book.bookname.bt.typename,pb.pubname.bc.name from 
tb_bookinfo book join tb_booktype bt on book.typeid=bt.id join tb_publishing pb on book.ISBN=pb.ISBN join tb_bookcase bc 
on book.bookcase=bc .id"): 


Sresult=mysql_fetch array($query): /应 用 外 联接 检索 图 书信 息 
?> 
过 // 省 略图 书信 息 标题 HTML 标 记 部 分 
<?php 
do{ // 应 用 do…while 循 环 语句 输出 查询 结果 


?> 
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<t> 
<td style= "padding:Spx:">&nbsp:<?php echo Sresult[barcode]:?></td> 
<td style= "padding:Spx:"><a hre 全 "book look.php?id=<?php echo Sresult[bookid]:?>"><2php echo 
$result[bookname]:?></a></td> 
<td style= "padding:Spx:">&nbsp:<?php echo $result[typename]:?></td> 
<td style="padding:5px:">&nbsp:<2?php echo $result[pubname]:?></td> 
<td style= "padding:5Spx:">&nbsp:<?php echo Sresult[name]:?></td> 
<td align="center"><a href="book_Modify.php?id=<?php echo Sresult[bookid]:?>"> 修 改 </a></td> 
<td align="center"><a href="book_del.php?id=<?php echo $result[bookid]:?>"> 删 除 </a></td> 
</a> 
<? 
}while(Sresult=mysql fetch_array(Squery)); //do…while 循 环 语句 结束 
?> 


名 注意 : 关于 join.…on 内 联接 语句 的 使 用 方法 参见 7.10.1 节 
TT.4 添加 图 书信 息 的 实现 过 过 程 


国 ”添加 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 系统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 在 该 
页 面 中 单 击 “ 添 加 图 书信 息 ” 超 链接 ， 进 入 到 添加 图 书信 息 页 面 。 添 加 图 书信 息 页 面 的 运行 结果 如 
图 7.25 所 示 。 


入 ) 本 让 周 思 馆 管理 系统 = 


DO KAO CIBRARY mama ovamm 


条 有 形 码 [666838999 
图 书 名 称 i 
图 HH 关 型 : | 
作 者 : | 
译 者 : Pestt 
出 版 社 : Re 入 本 
价 格 : %) 
页 玛 : 5 
书架 : [EE | 
EC 


CopyRight 日 2007 wr nrbccd com 言 林 rr 师范 大 学 图 书 演 
本 站 请 使 用 IE 6. 0 或 以 上 版 本 1024*768 为 最 佳 显示 效果 


图 7.25 添加 图 书信 息 页 面 的 运行 结果 
在 查看 图 书 列表 页 面 中 ， 设 置 “ 添 加 图 书信 息 ” 超 链接 的 代码 如 下 : 


> 
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例 程 15 ”代码 位 置 : 光盘 \TM\07\library\book.php 

<a href="book_add.php"> 添 加 图 书信 息 </a> 

添加 图 书信 息 页 面 主要 用 于 收集 输入 的 图 书信 息 以 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 
是 否 合法 。 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 7.4 所 示 。 


表 7.4 添加 图 书信 息 页 面 所 涉及 的 重要 表单 元 素 
名 称 元 素 类 型 


重要 属性 


forml form 


method="post" action= "book_okphp” 


typeld select 


isbn select 


bookcaseid select 


<?php include("Conn/conn.php"); 
$sql=mysql query("select * from tb_booktype"): 
$info=mysql fetch array($sq)); 
dof 

?> 

<option value="<?php echo $info[id]:?>"> 

<?php echo $info[typename]:?></option> 


<?php 
$sql2=mysql query("select * from tb_publishing"): 
Sinfo2=mysql_fetch_array($sql2); 
dof 
> 
<option value="<?php echo $info2[ISBN]:?>"> 
<?php echo $info2[pubname]:?></option> 
<?php }while($info2=mysql_fetch_array($sql2)):?> 
<2php 
$sql3=mysql_query("select * from tb_bookcase"); 
Sinfo3=mysql fetch_array($sql3): 
do{ 
> 
<option value="<?php echo $info3[id]:?>"> 
<?php echo $info3[name]:?></option> 
<?php }while($info3=mysql fetch_array($sqgl3)):?> 


图 书 类 型 


出 版 社 


书架 名 称 


operator hidden 


value="<?php echo $info3[name]:?>" 


操作 员 


Submit submit 


‘onClick="return check(form1)" 


“保存 ”按钮 


Submit2 button 


onClick="history.back(|:" 


“返回 ”按钮 


由 于 添加 图 书信 息 的 方法 同 添加 管理 员 信息 的 方法 类 似 ， 所 以 此 处 只 给 出 向 图 书信 息 表 中 插入 数 


据 的 SQL 语句 ， 详 细 代码 参见 光盘 。 向 图 书信 息 表 中 插入 数据 的 SQL 语句 如 下 : 


例 程 16 ”代码 位 置 : 光盘 \TMW7Wibrary\book ok php 


mysql query("insert into 


tb_bookinfo(barcode.bookName,.typeid.author.translator.ISBN.price.page.bookcase.inTime.operator )values('$barcode'.'$bookN 


345 
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ame''$typeid','$author.'$translator,$isbn''$price'"$page','S$bookcaseid'.'$inTime'S$operator)”): 
7.7.5 修改 图 书信 息 的 实现 过 程 


国 修改 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_ bookcase、tb_booktype、tb publishing 

管理 员 登 录 系 统 后 ， 在 导航 栏 中 单 击 “图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 单 击 
想 要 修改 的 图 书信 息 后 面 的 “修改 ” 超 链 接 ， 进 入 到 “修改 图 书信 息 ” 页 面 。 修 改 图 书信 息 页 面 的 运 
行 结果 如 图 7.26 所 示 。 


午 ) 醋 志 图 志 馆 管理 系统 


BO KAO DINIARY wwAse vam 


图 书 名 称 ; Pop 网 后 妨 程 目 学 手册 * 
图 书 类 到 ; 可 

作 者 ; 硬 天 下 表亲 素 、 记 中 华 

译 者 : rsoft 

出 版 社 

价 格 : ]6) 

页 码 : 

书架 : 


CopyRight 日 2007 mm_ nrbecd com 守 林 *** 岳 范 大 学 图 书馆 
本 站 请 合用 IE 6. 0 或 以 上 版 本 1024*766 为 景 佳 显示 效果 


图 7.26 修改 图 书信 息 页 面 的 运行 结果 

在 图 书信 息 列表 页 面 中 ， 添 加 “修改 ” 超 链 接 的 代码 如 下 : 

例 程 17 ”代码 位 置 : 光盘 \TMW7Nlibrary\book php 

<a hre 全 "book Modify php?id=<2php echo $result[bookid]:?>"> 修 改 </a> 

在 修改 图 书信 息 页 面 中 修改 图 书信 息 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信 息 到 数据 处 理 页 
book Modify_ okphp， 应 用 UPDATE 语句 将 修改 的 图 书信 息 保存 到 数据 表 tb_bookinfo 中 ， 并 弹出 “图 
书信 息 修 改 成 功 ! ”提示 信息 ， 将 页 面 重 定向 到 修改 图 书信 息 页 。 数 据 处 理 页 的 代码 如 下 : 

例 程 18 代码 位 置 : 光盘 \TM\07\library\book_Modify_ok.php 


<?php 

Session start(): // 初 始 化 session 变 量 
include("conn/conn.php"): // 连 接 数 据 库 文件 
S$bid=$_POST[bid]: // 获 取 图 书 id 号 
S$operator-$_ SESSION[admin name]: /获取 管理 员 名 称 


Es 
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Sbarcode=$_POST[barcode]: // 获 取 图 书 条 形 码 
$bookName=$ POST[bookName]: // 获 取 图 书 名 称 
Stypeid=$_POST[typeId]: // 获 取 图 书 类 型 id 号 
$author=$_POST[author]: /获取 图 书 作 者 

$translator-$ POST[translator]: // 获 取 图 书 译 者 

Sisbn=$_ POST[isbn]: // 获 取出 版 社 ISBN 
Sprice=$_POST[price]; /获取 图 书 单价 

S$page=$_ POST[page]: /获取 图 书页 码 
$bookcase=$_POST[bookcaseid]: // 获 取 图 书 书架 id 号 
SinTime=date("Y-m-d"): // 设 置 图 书 更 新 日 期 为 当前 日 期 


$query=mysql query("update tb_bookinfo set barcode='$barcode', bookName='$bookName' , typeid='$typeid', 
author='$author', translator=-'$translator, ISBN= Sisbn' , price='$price' , page='$page' , bookcase='$bookcaseid', 


inTime='$inTime' operator="$operator' where id=$bid"): // 更 新 数据 表 
echo "<script language='javascript>alert(' 图 书信 息 修改 成 功 !"):history.back 〇 :</script>"; 
?> 


7.7.6 删除 图 书信 息 的 实现 过 程 


国 。 删除 图 书信 息 使 用 的 数据 表 : tb_bookinfo 

在 查看 图 书 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 

例 程 19 代码 位 置 : 光盘 \TM\07\library\book.php 

<a href="book_del.php?id=<?php echo Sresult[bookid]:?>"> 删 除 </a> 

单 击 想 要 删除 的 图 书信 息 后 面 的 “删除 ” 超 链接 ， 提 交 表 单 信息 到 数据 处 理 页 book_del.php， 应 用 


DELETE 语句 将 指定 的 图 书信 息 从 数据 表 tb_bookinfo 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “图 书 
信息 删除 成 功 ! ”提示 信息 ， 并 将 页 面 重 定向 到 图 书信 息 列 表 页 面 。 数 据 处 理 页 的 代码 如 下 : 


例 程 20 ”代码 位 置 ; 光盘 \TMVWO7Nlibrary\book_del.php 


<?php 

include("conn/conn.php"): // 连 接 数 据 库 文件 
Sinfo_del=mysql_query("delete from tb_bookinfo where id=$_GET[id]"): // 删 除 指定 的 图 书信 息 

ifSinfo deD){ // 如 果 信息 删除 成 功 ， 则 弹出 提示 


echo "<script language='javascript>alert(' 图 书信 息 删除 成 功 !"):history.backO:</script> "; 
b 
es 


7.8 图 书 借 还 模块 设计 


7.8.1 图 书 借 还 模块 概述 


图 书 借 还 模块 主要 包括 图 书 借阅 、 图 书 续 借 、 图 书 归 还 、 图 书 档案 查询 、 图 书 借阅 查询 、 借 阅 到 
期 提醒 等 6 个 功能 。 在 图 书 借阅 模块 中 的 用 户 只 有 一 种 身份 ， 那 就 是 操作 员 ， 通 过 该 身份 可 以 进行 图 
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书 借 还 等 相关 操作 。 图 书 借 还 模块 的 用 例 图 如 图 7.27 所 示 。 


图 书 档案 查询 / 图 书 借阅 


图 书 借阅 查询 ) < 图 书 续 借 
借阅 到 期 提醒 操作 员 图 书 归 还 


图 7.27 图 书 借 还 模块 的 用 例 图 


7.8.2 图 书 借 还 模块 技术 分 析 


在 图 书 借 还 模块 中 涉及 到 的 数据 表 是 tb_borrow (图书 借阅 信息 表 ) 、tb_bookinfo (图 书信 息 表 ) 
和 tb _reader (读者 信息 表 ) ， 这 3 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 如 图 7.28 所 示 。 


tb_reader 
Dose i integer warigned Gk) 
国 integer (11) 《pk> mame varchar (20) 
arcode int sex varchar (4) 
boolknane varchar (70) barcode varchar (30) 
typeid integer unsigned FR_reader yocation varchar (50) 
tr var' es 0 IFK_book tb_borrow birthday 。 date 
translator varchar = = 了 T! har (10) 

和 re paperType varchar 

Ea 0) a i /a wee 
price float (8,2) 和 tel varchar (20) 
Be I a Gd bookid integer (10) <Ekl> 
bookease integer unsigned 性 pas Re 
En Fe BockTine dute createDate date 
operator varchar (30) 时 和 ey 
Pp ifback tinyint (1) Et 
del tinyint (1) typeid integer (11) 


7.28 图 书 借 还 管理 模块 各 表 间 关系 图 
全 注意 : 本 模块 主要 应 用 联接 语句 实现 多 表 查询 ， 关 于 联接 语句 的 详细 讲解 参见 7.10 节 。 


7.8.3 图 书 借阅 的 实现 过 程 


国 ”图 书 借阅 使 用 的 数据 表 : tb_borow、tb_bookinfo、tb_reader 
管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 借阅 ”菜单 项 ， 进 入 到 图 书 借阅 页 面 ， 在 该 页 面 中 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 〈 如 123456789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 将 其 显示 在 页 面 中 ， 此 时 输 
入 图 书 的 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ” 按 钮 ， 借 阅 指定 的 图 书 ， 运 行 结果 如 图 7.29 所 示 。 
[DD 说 明 : 当 读 者 借阅 图 书 完毕 后 ， 操 作 员 通过 单 击 “ 完 成 借阅 ”按钮 ， 将 重新 载 入 图 书 借阅 页 面 ， 
当前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 进 行 下 一 个 读者 借阅 图 书 操作 。 
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情 记 图 书馆 管理 系统 


NO KAO [iBRARY MAnAGe syYsTEw 


当 和 位 尖 : 朋 书 梭 正 》 有 图书 协 同 》》 


有 作 国 乌 供 阅 


CopyRight 图 2007 ww arbccd com 计 林 +*+* 师 范 大 学 图 书馆 
本 站 请 使 用 IE 6 0 或 以 上 版 本 1024*768 为 最 住 显示 效果 


7.29 图 书 借阅 页 面 的 运行 结果 


图 书 借阅 页 面 总 体 上 可 以 分 为 两 个 部 分 : 一 部 分 用 于 查询 并 显示 读者 信息 ; 另 一 部 分 用 于 显示 读 
者 的 借阅 信息 和 添加 读者 借阅 信息 。 图 书 借阅 页 面 在 Dreamweaver 中 的 设计 效果 如 图 7.30 所 示 。 


| 还 件 类 型 : FRR echo SinFo[lpwperT 证 件 号 码 : Fig echo Finfofpaper 基 河 异 数量 : [Si echo FinEo 册 


7.30 图 书 借阅 页 面 的 设计 效果 


在 进行 图 书 借阅 时 ， 系 统 要 求 每 个 读者 只 能 同时 借阅 一 定数 量 的 图 书 ， 并 且 该 数量 由 读者 类 型 表 
tb readerType 中 的 可 借 数量 number 决定 ， 所 以 笔者 编写 了 自 定义 的 checkbookO 函 数 ， 用 于 判断 当前 
选择 的 读者 是 否 还 可 以 借阅 新 的 图 书 ， 同 时 该 函数 还 具有 判断 输入 读者 条 形 码 或 图 书 名 称 文本 框 是 否 
为 空 的 功能 。 代 码 如 下 : 


例 程 21 代码 位 置 : 光盘 \TMW2\bookBorowphp 
<script language="javascript"> 


function checkbook(form){ // 自 定义 一 个 JavaScript 函 数 checkbookO) 


这 form barcode value 一 ""){ 
alert(" 请 输入 读者 条 形 码 !"):form barcode focusO:retum: 


) 
if(form.inputkey.value—""){ 


/如 果 读 者 条 形 码 为 空 
/弹出 提示 ， 焦 点 返回 到 条 形 码 文本 杠 


o 


/如 果 图 书 查询 文本 框 的 值 为 空 
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alert(" 请 输入 查询 关键 字 !"):form .inputkey .focus(:return:; // 弹 出 提示 ， 焦 点 返回 到 图 书 查询 文本 框 
} 


if(form.number.value-form.borrowNumber.value<=0){ // 如 果 图 书 的 借阅 数量 超过 了 可 借 数 量 
alert(" 您 不 能 再 借阅 其 他 图 书 了 !"):return: /弹出 提示 信息 
form.submit(): /提交 表单 
} 
</script> 


技巧 :在 JavaScript 中 比较 两 个 数值 型 文本 框 的 值 时 ， 不 使 用 运算 符 “--”， 而 是 先 将 这 两 个 值 
相 减 ， 再 判断 其 结果 。 


检索 读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 的 SQL 语句 如 下 : 


例 程 22 ”代码 位 置 : 光盘 \TMWO7Nibrary\bookborrowphp 

$sql=mysql_query("select r.*,t.name as typename.Lnumber from tb reader r left join tb_readerType t on r.typeid=t.id 
where r.barcode='$barcode”™): 

$info=mysql_fetch_array($sql); /检索 读者 信息 


获取 读者 借阅 信息 的 SQL 语句 如 下 : 


例 程 23 ”代码 位 置 : 光盘 \TM\07\library\bookborrow.php 
$sqll=mysql_query("select r.*,borr.borrowTime,borr.backTime,book.bookname,book.price.pub.pubname.bc.name as 
bookcase from tb_borrow as borr join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as pub on 
book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id join tb_reader as T on boir.readerid=r.id where 
borr.readerid='$readerid' and borr.ifback=0"): 
Sinfol=mysql fetch_array($sql1): /检索 读者 的 借阅 信息 
S$borrowNumber=mysql num rows($sql1): /获取 结果 集中 行 的 数目 


在 “图 书 条 形 码 ”/“ 图 书 名 称 ” 文 本 框 中 输入 图 书 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ” 按 钮 ， 检 
索 图 书信 息 是 否 存 在 ， 如 果 不 存在 ， 则 向 图 书 借阅 信息 表 中 添加 该 读者 的 图 书 的 借阅 记录 ， 完 成 图 书 
殿 阅 操作 ， 否 则 ， 弹 出 该 书 不 能 被 同一 读者 重复 借阅 的 提示 信息 。 图 书 借阅 的 具体 代码 如 下 : 

例 程 24 ”代码 位 置 : 光盘 \TMW7Nlibrary\bookBorrowphp 


<?php 
if($_POST[inputkey]!=""){ /如 果 “ 图 书 条 形 码 ”/“ 图 书 名 称 ” 后 的 文本 框 不 为 空 
S$f-$ POST: /获取 用 户 选 择 的 条 件 值 
@ $inputkey=trim($ POST[inputkey]): // 获 取 用 户 输入 的 查询 关键 字 
$barcode=$ POST[barcode]: // 获 取 读 者 的 条 形 码 
S$readerid=$ POST[readerid]: /获取 读者 id 号 
@ $borowTime=date('Y-m-d’"); // 图 书 的 借阅 时 间 为 系统 当前 时 间 


©@ S$backTime=date("Y-m-d",(time()}+3600*24*30)); // 归 还 图 书 日 期 为 当前 期 日 期 +30 天 期 限 
$query=mysql _ query("select * from tb_bookinfo where $f='$inputkey"™): 


$result=mysql_fetch_array($query): // 检 索 图 书信 息 是 否 存 在 
这 Sresult 一 包 lsej{ /如 果 读 者 借阅 的 图 书 不 存在 ， 那 么 弹出 提示 信息 


echo "<script languagejavascript>alert(' 该 图 书 不 存在 ! 
"):window.location.href='bookBorrow.php?barcode=S$barcode': </script>"; 
; 
else{ // 检 索 该 读者 所 借阅 的 图 书 是 否 与 再 借 图 书 重复 
Squeryl=mysql query("select 1.*,borr.borrowTime,borr.backTime.book.bookname.book.price,pub.pubname.bc.name 
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as bookcase from tb borow as borr join tb_reader as T on borr.readerid=r.id join tb_ bookinfo as book on book.id=borr.bookid 
join tb publishing as pub on bookISBN=pub.ISBN join tb bookcase as bc on bookbookcase=bc.id where 
borr.bookid=$result[id] and borr.readerid=$readerid and ifback=0"); 
Sresultl=mysql fetch array($query1): 
if($resultl=—true){ // 如 果 所 借 图 书 已 被 该 读者 借阅 ， 那 么 提示 不 能 重复 借阅 
echo "<script languagejavascript>alert( 该 图 书 已 经 借阅 ! 
"):window.location.href='bookBorrow.php?barcode=$barcode':</script>": 
! 
else{ // 否 则 ， 完 成 图 书 借阅 操作 ， 并 弹出 借阅 成 功 提示 信息 
$bookid=S$result[id]; // 将 读者 id 号 赋 给 一 变量 
mysql_query("insert into 
tb_borrow(readerid,bookid,borrowTime,backTime,operator,ifback)values('$readerid','$bookid','$borrowTime','$backTime',$_S 
ESSION[admin name]',.0)"): /向 借阅 信息 表 中 添加 一 条 借阅 信息 
echo "<script language='javascript>alert( 图 书 借阅 操作 成 功 ! 
):window.location.href='bookBorrow.php?barcode=$barcode';</script>"; 
} 
} 


?> 
< 人 代码 贴 二 
@ tim0: 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 ， 以 达到 精确 查询 。 
@ date(Y-m-d): 获取 系统 的 当前 日 期 为 图 书 借 阅 的 日 期 ， 并 格式 化 日 期 格式 。 
@ date("Y-m-d",(time()+3600*24*30)): 归还 图 书 日 期 = 当前 期 日 期 130 天 期 限 ,当前 日 期 时 间 蕉 应 用 time0 函 数 获取 ， 
30 天 期 限 的 时 间 鹤 等 于 3600 秒 x24 小 时 x30 天 ， 并 通过 date0 函 数 格 式 化 为 指定 日 期 格式 。 


7.8.4 图 书 续 借 的 实现 过 程 


国 ”图 书 续 借 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 续 借 ” 菜 单项 ， 进 入 到 图 书 续 借 页 面 。 在 该 页 面 中 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 123456789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 显示 在 页 面 中 ， 此 时 
单 击 “ 续 借 ” 超 链接 , 即 可 续 借 指 定 图 书 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ) 。 
图 书 续 借 页 面 的 运行 结果 如 图 7.31 所 示 。 


[加 说 明 : 当 读 者 续 借 完 图 书后 ， 操 作 员 通 过 单 击 “ 完 成 续 借 ”按钮 ， 将 重新 载 入 图 书 续 借 页 面 ， 当 
前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 进 行 下 一 个 读者 续 借 图 书 操作 。 
图 书 续 借 页 面 的 设计 方法 同 图 书 借阅 类 似 ， 所 不 同 的 是 ， 在 图 书 续 借 页 面 中 没有 添加 借阅 图 书 的 
功能 ， 而 是 添加 了 “ 续 借 ” 超 链接 。 图 书 续 借 页 面 在 Dreamweaver 中 的 设计 效果 如 图 7.32 所 示 。 
单 击 “ 续 借 ” 超 链接 时 ， 还 需要 将 读者 条 形 码 、 借 阅 id 号 和 图 书 归还 时 间 一 同 传递 到 图 书 续 借 的 
处 理 页 borrow_oncemore.php 中。 代码 如 下 : 
例 程 25 ”代码 位 置 : 光盘 \TM\07\library\bookRenew.php 


<a href="borrow_oncemore.php?barcode=<?php echo $infofbarcode]:?>&borrid=<?php echo 
S$info[borid]:?>&backTime=<?php echo $infof[backTime]:?>"> 续 借 </a> 
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7.31 图 书 续 借 页 面 的 运行 结果 


当前 位 置 : 图 书 异 还 》 图 书 综 僧 >>》 


图 书 名 称 借阅 时 间 Ex 出 版 柱 定价 多 ) 这 码 贡 | 
尖 吕 ED 吕 EE Ey Li 


检索 读者 信息 和 读者 借阅 信息 的 SQL 语句 如 下 : 
例 程 26 ”代码 位 置 : 光盘 \TMVW7Nibrary\bookRenew.php 


$sql=mysql query("select borr.id as borrid.borrborrowTime.borr.backTime.borrifbackT.*.Lname as 
typename.tLDumber.book.bookname.book.price.pub.pubname,bc.name as bookcase from tb_borrow as borr join tb_reader r on 
borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as 
pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id where r.barcode='$barcode' and 
borr.ifback=0"): 


Sinfo=mysql_fetch_array($sq)): /检索 读者 信息 和 借阅 信息 


单 击 “ 续 借 ” 超 链接 ， 提 交 到 数据 处 理 页 borrow_oncemore.php， 主 要 用 于 完成 图 书 的 续 借 功能 ， 
主要 通过 更 改 图 书 的 归还 日 期 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ， 续 借 日 期 
的 具体 算法 ， 参 见 7.10.2 节 的 详细 讲解 ) 实现 。 数 据 处 理 页 的 代码 如 下 : 

例 程 27 代码 位 置 光盘 \TM\07\library\borrow_oncemore.php 


<2php 
session start(): // 初 始 化 session 变 量 


> 
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include("conn/conn.php"): // 连 接 数 据 库 文件 
Sbarcode=$_GET[barcodel: // 获 取 图 书 条 形 码 
Snew=$_GET[backTime]: // 获 取 图 书 归 还 时 间 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 期 日 转化 为 时 间 截 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 

SnewbackTime=date("Y-m-d".(mktime(0, 0. 0, substr(Snew.5.2), substr(Snew.8.2), substr(S$new,0.4))+3600*24*30)): 

Sborrid=$_GET[borrid]: // 获 取 续 借 图 书 的 id 号 

mysql_ query("update tb_borrow set backTime='$newbackTime',ifback=0,operator='$_SESSION[admin namel' where 
id=$borrid"): 

echo "<script language='javascript>alert(' 图 书 续 借 操作 成 功 ! 
"):window.location.href='bookRenew.php?barcode=$barcode';</script>"; // 弹 出 图 书 续 借 成 功 的 提示 信息 

?> 


7.8.5 图 书 归还 的 实现 过 程 


国 ”图 书 归 还 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 归 还 ”菜单 项 ， 进 入 到 图 书 归 还 页 面 。 在 该 页 面 中 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 123456789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 输出 到 浏览 器 ， 此 时 
单 击 “ 归 还 ” 超 链 接 ， 即 可 将 指定 图 书 归 还 。 图 书 归还 页 面 的 运行 结果 如 图 7.33 所 示 。 
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图 7.33 图 书 归 还 页 面 的 运行 结果 


图 书 归还 页 面 的 设计 方法 同 图 书 续 借 类 似 ， 所 不 同 的 是 ， 将 图 书 续 借 页 面 中 的 “ 续 借 ” 超 链接 转 
化 为 “归还 ” 超 链 接 。 在 单 击 “归还 ” 超 链接 时 ， 也 需要 将 读者 条 形 码 和 借阅 id 号 一 同 传递 到 图 书 归 
还 处 理 页 ， 代 码 如 下 : 

例 程 28 ”代码 位 置 光盘 \TM\07\library\bookBack.php 

<a href="bookBack_ok php?borid=<?php echo $infofborrid]:?>&barcode=<?php echo $infofbarcode]:?>"> 归 还 </a> 


检索 读者 信息 及 读者 借阅 信息 的 SQL 语句 如 下 : 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


例 程 29 ”代码 位 置 :光盘 \TM\07\library\bookBack.php 
$sql=mysql query("select borr.id as borrid,borr.borrowTime.borr.backTime,borr.ifback.r.*.t.name as 
typename,t.number,book.bookname,book.price.pub.pubname,bc.name as bookcase from tb_borrow as borr join tb_readerr on 
borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as 
pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book bookcase=bc.id where 1.barcode='$barcode' and 
borrifback=0"): 
$info=mysql fetch_ array(SsqD): /检索 读者 信息 及 该 读者 的 借阅 信息 


单 击 “ 归 还 ” 超 链接 ， 即 可 将 指定 图 书 归 还 。 数 据 处 理 页 的 代码 如 下 : 
例 程 30 ”代码 位 置 : 光盘 \TMWO7Nibrary\bookBack okphp 


<2php 

session_start(): // 初 始 化 session 变 量 

include("conn/conn.php"): // 连 接 数 据 库 文件 

S$backTime=date("Y-m-d"): // 归 还 图 书 日 期 

S$borrid=$_GET[borrid]: /获取 读者 的 id 号 

mysql_query("update tb_borrow setbackTime='$backTime'ifback=1.operator=-'$_ SESSION[admin _ name] where 
id=-$borid"); /更 新 读者 的 借阅 信息 


echo "<script language='javascript>alert(' 图 书 归 还 操作 成 功 ! 
"):window.location.href='bookBack.php?barcode=$barcode':</script>": /弹出 图 书 归 还 成 功 的 提示 信息 
?> 


7.8.6 ”图 书 借阅 查询 的 实现 过 程 


国 图书 借阅 查询 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “ 系 统 查询 ”/“ 图书 借阅 查询 ”菜单 项 ， 进 入 到 图 书 借阅 查询 页 面 。 图 书 借 
阅 查 询 页 面 的 运行 结果 如 图 7.34 所 示 。 在 该 页 面 中 可 以 按 指定 的 字段 或 某 一 时 间 段 进行 查询 ， 同 时 还 
可 以 实现 按 指 定 字 段 及 时 间 段 进行 综合 条 件 查 询 。 
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图 7.34 图 书 借阅 查询 页 面 的 运行 结果 
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图 书 借阅 查询 页 面 主要 用 于 收集 查询 条 件 和 显示 查询 结果 , 并 通过 自 定义 的 JavaScript 函数 验证 输 
入 的 查询 条 件 是 否 合法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 7.5 所 示 。 


表 7.5 图 书 借 阅 查询 页 面 所 涉及 的 表单 元 素 


元 素 类 型 重要 属性 含义 
form method="post" action="" 表单 
checkbox value="a" 请 选择 查询 依据 
checkbox value="b" 借阅 时 间 
<option value="k.barcode" > 图 书 条 形 码 </option> 
<option value="k.bookname"> 图 书 名 称 </option> 


<option value='"r.barcode"> 读 者 条 形 码 </option> 查询 字段 

<option value="r.name"> 读 者 名 称 </option> 

size="50" 关键 字 

id="sdate" 开始 日 期 

id="edate" 结束 日 期 
“查询 ”按钮 


在 图 书 借阅 查询 页 面 中 ， 指 定 查 询 条件 后 , 提交 表单 信息 到 当前 页 。 首 先 获取 表单 元 素 复 选 框 flag 
的 值 ， 然 后 根据 flag 的 值 组 合 查 询 字 符 串 。 

如 果 flagl 的 值 等 于 a， 那 么 按 指 定 的 字段 检索 图 书 借阅 信息 ; 如 果 flag2 的 值 等 于 b， 那 么 按 指定 
的 时 间 段 检索 图 书 借阅 信息 ; 如 果 flag1 的 值 等 于 a， 并且 flag2 的 值 等 于 b， 那 么 按 以 上 两 个 条 件 的 综 
合 条 件 检索 图 书 借阅 信息 ， 并 将 查询 结果 输出 到 浏览 器 。 具 体 代码 如 下 : 

例 程 31 代码 位 置 : 光盘 VTMWO7Nibrary\borrowQueryphp 

<? 

a // 连 接 数 据 库 文件 


$sql=mysql_query("select b.borrowTime,b.backTime.b.ifbackrbarcode as 
Teaderbarcode.r.name,k.id,k.barcode,k.bookname from tb_borrow b join tb_readerr on breaderid=r.id join tb_bookinfo k on 


b.bookid=k.id"): // 查 询 图 书 借阅 信息 
if($_POST[Submit]!=""){ // 如 果 提 交 了 表单 ， 则 执行 以 下 操作 

$f=$_POST[f]: // 获 取 操 作 员 选择 的 查询 条 件 
S$keyl=$_ POST[keyl]: // 获 取 查询 关键 字 
S$sdate=$ POST[sdate]: // 获 取 借阅 的 起 始 日 期 
Sedate=$_POST[edate]: // 获 取 借 阅 的 结束 日 期 
Sflagl=$_POST[flag1]: // 获 取 按 指定 条 件 查 询 的 复 选 框 值 
Sflag2=$_POST[flag2]: // 获 取 按 日 期 查询 的 复 选 框 值 
if(Sflag1—"a"){ // 如 果 按 指定 条 件 查询 ， 则 执行 以 下 语句 


$sql=mysql query("selectb.borrowTime.b.backTime.b.ifbackrbarcode as 
Teaderbarcode.Tnamek.idk.barcode.kbookname from tb borrow b join tb_reader r on breaderid=r.id join tb_bookinfo k on 
b.bookid=k.id where $f like '%S$key1%"): 


b 
if(Sflag2—"b"){ /如 果 按 时 间 段 查询 ， 则 执行 以 下 语句 
S$sql=mysql_query("select b.borrowTime.b.backTime.b.ifback.r.barcode as 
readerbarcode,r.name,k.id,k.barcode.k.bookname from tb_borrow b join tb_reader r on b.readerid=.id join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate and '$edate"™): 
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b 
if(Sflag1l—"a" && Sflag2—"b"){ // 如 果 按 综合 条 件 查 询 ， 则 执行 以 下 语句 
$sql=mysql_query("select b.borrowTime,b.backTime.b.ifback.r.barcode as 
readerbarcode.r.name,k.id,k.barcode.k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate' and $f like '%S$key1%"): 
! 
} 
Sresult=mysql fetch_ array($sqD); /检索 查询 结果 
这 Sresult 一 false){ /如果 查询 结果 不 存在 ， 则 弹出 提示 信息 
?> 
<table width="100%" height="30" border="0" cellpadding="0" cellspacing="0"> 
<t> 
<td height="36" align="center"> 暂 无 图 书 借阅 信息 ! </td> 
</tr> 
</table> 
<?php 
} 
else{ // 否 则 ， 输 出 图 书 借阅 信息 
?> 
<table width="723" border="1" cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" bordercolordark="#D2E3E6" 
bordercolorlight="#FFFFFF"> 
<tr align="center" bgcolor='"#DOE9F8"> 
<td width="13%"> 图 书 条 形 码 </td> 
<td width="27%"> 图 书 名称 </td> 
<td width="15%"> 读 者 条 形 码 </td> 
1%"> 读 者 名 称 </td> 
"13%"> 借 阅 时 间 </td> 
<td width="11%"> 归 还 时 间 </td> 
<td width="10%"> 是 否 归 还 </td> 
</t> 
<?php 
dof 
if(Sresult[ifback]=—"0"){ // 如 果 “ 是 否 归还 ”等 于 0， 则 输出 “未 归还 ” 
Sifbackstr=" 未 归还 ": 


} 
else{ // 如 果 “ 是 否 归还 ”等 于 1， 则 输出 “已 归还 ” 
Sifbackstr=" 已 归还 ": 
} 
?> 
站 中 中 中 下 站 字 中 中字 守 站 中 下 中 中 中 中 站 束 训 本 中 中 下 字 中 站 本本 站 克 出 符合 查询 条 件 的 记录 相让 本 相 让 中 中 宙 本 机 下 机 事 事 可 可 机 事 可 可 机 事 字 下 事 字 吕 */ 
<t> 
<td style="padding:Spx:">&nbsp:<?php echo $result[barcode]:?></td> 
<td style= "padding:Spx:"><a href="book look.php?id=<?php echo $result[id]: ?>"><?php echo 
$result[bookname]:?></a></td> 
<td style= "padding:5Spx:">&nbsp:<?php echo $result[readerbarcode]:?></td> 
<td style="padding:Spx:">&nbsp:<?php echo $result[name]:?></td> 
<td style="padding:Spx:">&nbsp:<?php echo $result[borrowTime]:?></td> 
<td style="padding:5px:">é&nbsp:<?php echo $result[backTime]:?></td> 
<td style="padding:Spx:">&nbsp:<?php echo $ifbackstr:?></td> 
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</t> 
/本 可 于 于 于 于 二 于 可 于 于 素 于 宙 束 于 于 环球 于 可 于 定 守 本 于 可 宙 守 于 本 村 可 训 下 于 村 可 训 本 于 本 可可 于 村 于 字 本 机 可 本 于 字 于 机 机 本 于 机 机 可 本 于 字 机 宁可 可 可 村 可 束 束 末末 宁可 束 可 训 站 / 


<?php 
}while(Sresult=mysql fetch array(SsqD); 
} 


7.8.7 ”单元 测试 


在 开发 完 图 书 借 还 模块 后 ， 对 该 模块 进行 了 单元 测试 。 当 操作 员 按 指定 字段 进行 条 件 查 询 时 ， 能 
够 实现 图 书 借阅 查询 ， 当 操作 员 按 指定 的 时 间 段 进行 查询 时 ， 也 能 够 实现 图 书 借阅 查询 ， 但 是 当 操作 
员 按 以 上 两 个 条 件 同 时 进行 查询 时 ， 则 不 能 实现 图 书 借阅 查询 。 

图 书 借 还 模块 的 表单 元 素 代码 如 下 : 


<input name="flag" type="checkbox" class="noborder" value="a" checked> 
<input name="flag" type="checkbox" class="noborder" id="flag" value="b"> 


完成 图 书 借阅 查询 的 SQL 语句 如 下 : 


if(Sflag—"a"){ 
S$sql=mysql_query("select b.borrowTime.b.backTime.b.ifback.r.barcode as 
Teaderbarcode,r.name,k.id,k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on 
b.bookid=k.id where $f like '%$key1%"): 
9 
if(Sfiag—"b"){ 
$sql=mysql query("select b.borrowTime,b.backTime.b.ifback,r.barcode as 
readerbarcode,r.name,k.id,k.barcode.k.bookname from tb_borrow b join tb_reader r on breaderid=r.id join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate'"); 
} 
if(Sflag—"a" && Sflag—"b"){ 
$sql=mysql_query("select b.borrowTime.b.backTime.b.ifback.r.barcode as 
Teaderbarcode,r.name,k.id,k.barcode.k.bookname from tb_borrow b join tb_ reader r on brTeaderid=r.id join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"): 
} 


为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo0 语 句 对 $flag 的 值 进行 输出 ， 输 出 的 结果 显示 $flag 等 于 b， 
不 执行 $flag 等 于 a。 这 充分 说 明 在 对 同名 复 选 框 进行 查询 时 ， 它 只 能 获取 最 后 一 个 复 选 框 的 值 。 

解决 该 问题 的 方法 是 将 两 个 复 选 框 分 别 定义 为 两 个 不 同 的 名 称 ， 然 后 赋予 不 同 的 值 ， 再 进行 判断 ， 
就 可 以 实现 按 指定 字段 、 指 定 的 条 件 进行 综合 条 件 查询 了 。 

因此 将 表单 中 的 HTML 标记 中 的 hag 进行 重 命名 ， 代 码 如 下 ;: 


<input name="flag1" type= "checkbox" class="noborder" value="a" checked> 
<input name="flag2" type="checkbox" class="noborder" id="flag”" value="b"> 


修改 后 的 完成 图 书 借阅 查询 的 SQL 语句 如 下 : 


ismagl 一 "af 
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$sql=mysql_query("select b.borrowTime.b.backTime.b.ifback.r.barcode as 
Teaderbarcode.r.name.k.id.k.barcode.k.bookname from tb borrow b join tb reader r on breaderid=r.id join tb bookinfo k on 
b.bookid=k.id where $f like '%$key1%"); 

} 
if(Sflag2—"b"){ 

$sql=mysql_query("select b.borrowTime.b.backTime,b.ifback,r.barcode as 
Teaderbarcode,r.name,k.id,k.barcode,k.bookname from tb_borrow b join tb reader r on breaderid=rid join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate”"™"): 

} 
if(Sflag1l—"a" &e& Sflag2—"b"){ 

S$sql=mysql_query("select b.borrowTime,b.backTime.b.ifback.r.barcode as 
Teaderbarcode,r.name,k.id,k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 

} 


7.9 开发 技巧 与 难点 分 析 


7.9.1 如 何 自动 计算 图 书 归 还 日 其 


在 图 书馆 管理 系统 中 会 遇 到 这 样 的 问题 : 在 借阅 图 书 时 ， 和 需要 自动 计算 图 书 的 归还 日 期 。 

1， 图书 归 还 日 期 

根据 图 书馆 还 书 的 规律 一 般 都 以 30 天 为 一 个 期 限 ， 因 此 在 图 书 归还 时 ， 可 以 设置 一 个 固定 的 值 ， 
即 30 天 。 计 算 归 还 日 期 的 方法 如 下 : 

图 书 归 还 日 期 =“ 系 统 当前 日 期 ”+“ 借 阅 天 数 固定 值 30 天 ” 

自动 计算 图 书 归 还 日 期 的 具体 代码 如 下 : 

date("Y-m-d",(time()+3600*24*30)) // 图 书 归 还 日 期 

2. 续 借 图 书 归 还 日 期 

续 借 图 书 归 还 日 期 是 在 原来 数据 库 保 存 该 图 书 归 还 日 期 (这 个 日 期 是 不 固定 的 ) 的 基础 上 再 次 借 
阅 所 计算 的 时 间 , 它 是 需要 根据 数据 表 中 保存 归还 日 期 来 计算 的 。 计 算 图 书 续 借 归还 日 期 的 方法 如 下 : 

续 借 图 书 归 还 日 期 =“ 所 借 图 书 在 数据 表 中 的 归 回 日 期 ”+ “借阅 天 数 固定 值 30 天 ” 

首先 应 用 substrO0 函 数 分 别 取出 所 借 图 书 在 数据 表 中 原 定 的 归还 日 期 “月 ”、“ 日 ”、“ 年 ”， 然 
后 应 用 mktimeO 函 数 计算 出 归还 日 期 的 时 间 戳 ， 最 后 应 用 date0 函 数 格式 化 日 期 为 “YYYY-MM-DD?” 
格式 。 自 动 计算 续 借 图 书 归 还 日 期 的 代码 如 下 : 

Snew=$_GET[backTime]: /获取 传递 过 来 的 该 图 书 在 数据 表 中 的 归还 日 期 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 期 日 转化 为 时 间 截 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
date("Y-m-d",(mktime(0, 0, 0. substr($new,5.2). substr(Snew.8.2). substr(Snew.0.4))+3600*24*30)): 
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7.9.2 ”如 何 对 图 书 借阅 信息 进行 统计 排行 


在 图 书馆 管理 系统 的 首页 中 ， 提 供 了 显示 图 书 借阅 排行 榜 功 能 。 要 实现 该 功能 ， 最 重要 的 是 如 何 
获取 统计 排行 信息 , 这 可 以 通过 一 条 SQL 语句 实现 。 本 系统 中 实现 对 图 书 借阅 信息 进行 统计 排行 的 SQL 
语句 如 下 : 

select * fiom (select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr join (select b.*,c.name as 
bookcasename,p.pubname.t.typename fiom tb_bookinfo b left join tb_bookcase ¢ on b.bookcase=c.id join tb_publishing p on 
b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book on borr.bookid=book.id order by borr.degree desc 
limit 10 

下 面 将 对 该 SQL 语句 进行 分 析 : 

(1) 对 图 书 借阅 信息 表 进 行 分 组 并 统计 每 本 图 书 的 借阅 次 数 ， 然 后 使 用 as 为 其 指定 别名 为 borr， 
代码 如 下 : 


(select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr 


(2) 使 用 左 联接 查询 出 图 书 的 完整 信息 ， 然 后 使 用 as 为 其 指定 别名 为 book， 代 码 如 下 : 


(select b.*,c.name as bookcasename,p.pubname,t.typename from tb_bookinfo b left join tb_bookcase c on b.bookcase=c.id 
join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book 


(3) 使 用 join on 语句 将 borr 和 book 连接 起 来 ， 再 对 其 按 统计 的 借阅 次 数 degree 进行 降序 排序 ， 
并 使 用 limit 子 句 限制 返回 的 行 数 。 


7.10 ”联接 语句 技术 专题 


在 实际 网 站 开发 过 程 中 ， 经 常 需要 从 多 个 表 中 查询 信息 ， 在 MySQL 数据 库 中 可 以 通过 连接 的 方 
式 实现 多 表 查 询 ， 连 接 方式 分 为 内 联接 和 外 联接 两 种 。 下 面 对 这 两 种 连接 方式 进行 详细 的 讲解 。 


7.10.1 内 联接 语句 


inner join 即 内 联接 查询 方式 ， 是 程序 开发 中 常用 的 连接 方式 。 内 联接 称 为 相等 联接 。 它 返回 两 个 
表 中 的 所 有 列 ， 但 只 返回 在 联接 列 中 具有 相等 值 的 行 。 内 联接 查询 的 语法 格式 如 下 : 
select fieldlist 


from tablel [inner] join table2 
on tablel.column=table2.column 


参数 说 明 : 


回 fieldlist， 要 查询 的 字段 列表 。 
回 tablel、table2: 为 要 连接 的 表 名 。 
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回 inner: 可 选项 ， 表 示 表 之 间 的 连接 方式 为 内 联接 。 

on tablel.column1=table2.column2: 用 于 指明 表 tablel 和 表 table2 之 间 的 连接 条 件 。 

下 面 通过 内 联接 方式 实现 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 ， 代 码 如 下 : 

$sql=mysql query("select tb yg.userid.tb yg.name,tb yg.sex.tb yg.age,tb yg.tel.tb yg.bm.tb yg info.gz from tb yg 
inner join tb_ yg _info on tb yg.userid=tb yg _info.ygid"); 

Sinfo=mysql fetch array($sq)): 


7.10.2 ”外 联接 语句 


内 联接 返回 的 是 两 个 表 中 符合 条 件 的 数据 ， 而 外 联接 返回 部 分 或 全 部 匹配 行 ， 这 主要 取决 与 所 建 
立 的 外 联接 的 类 型 。 外 联接 分 为 左 外 联接 和 右 外 联接 ， 下 面 对 两 个 外 联接 的 使 用 方法 进行 详细 讲解 。 
1. 左 外 联接 (left outer join) 
左 联接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 查询 条 件 及 右 表 中 所 有 满足 连接 条 件 的 行 。 在 
MySQL 数据 库 中 使 用 左 联接 的 语法 格式 如 下 : 
select field 1[field2…] 
from tablel left [outer] join table2 
on join condition 
[where search_condition] 
参数 说 明 : 
left outer join: 表示 表 之 间 通 过 左 联接 方式 相互 连接 ， 也 可 以 简写 成 left join。 
on join_condition: 指 多 表 建 立 连 接 所 使 用 的 连接 条 件 。 
where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 
下 面 通过 左 外 联接 的 方式 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 ， 代 码 如 下 : 
$sql=mysql_query("select * from tb_yg left outer join tb_yg_info on tb_yg.userid=tb_yg_info.ygid ",$conn); 
Sinfo=mysql fetch array($sq)):; 
2. 右 外 联接 (right outer join) 
右 联 接 返 回 的 查询 结果 包含 左 表 中 的 所 有 符合 连接 条 件 以 及 右 表 中 所 有 满足 查询 条 件 的 行 。 在 
MySQL 数据 库 中 使 用 右 联接 的 语法 格式 如 下 : 


select field 1[field2…] from tablel right [outer] join table2 on join_condition [where search_condition] 


参数 说 明 : 

回 right outer join: 表示 表 之 间 通 过 右 联 接 方 式 相互 连接 ， 也 可 以 简写 成 right join。 
outer: 可 选项 ， 表 示 表 之 间 的 联接 方式 为 完全 联接 。 

回 on join_condition: 指 多 表 建 立 联 接 所 使 用 的 连接 条 件 。 

加 ”where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 

下 面 通过 右 外 联接 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 ， 代 码 如 下 : 


S$sql=mysql_query("select * from tb yg right outer join tb yg info on tb yg.userid=tb yg info.ygid "); 
S$info=mysql fetch array($sqD): 
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7.11 本 章 总 结 


本 章 运用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 图 书馆 管理 系统 引导 读者 深入 了 解 系统 的 开发 流 
程 。 在 这 个 系统 的 实现 过 程 中 ， 除 了 应 用 一 些 基 本 的 PHP 技术 之 外 ， 还 涉及 一 些 独特 的 技术 细节 ， 如 
权限 设置 、 多 表 查 询 技术 等 需要 读者 掌握 ， 并 能 在 实际 的 操作 中 灵活 应 用 ， 举 一 反 三 。 
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第 章 


博客 管理 系统 


(Apache+PHP+phpMyAdmin+MYySQL 5.0 实现 ) 
( 铝 " 视频 讲解 : 1 小 时 16 分钟 ) 


如 果 有 人 问 “ 什 么 是 Web2.0? ”， 丽 怕 没 几 个 人 能 说 得 清楚 ， 但 是 如 果 问 什 
么 是 博客 ， 哪 怕 是 从 不 上 网 的 人 也 是 耳熟能详 的 了 。 博 客 正 是 Web2.0 概念 中 重要 
的 组 成 部 分 之 一 (大 家 熟知 的 还 包括 IM 即时 通 和 R55 阅读 器 ) 。 

Blog (博客 ) ， 全 名 Weblog， 后 来 简写 为 Blog。Blogger 就 是 写 Blog 的 人 ， 
习惯 于 在 网 上 写 出 日 记 、 发 布 个 人 照片 、 展 示 个 性 自我 的 用 户 群体 。 对 于 
Blog/Blogger 的 中 文 名 称 ， 有 翻译 成 “博客 ”， 也 有 翻译 为 “网 志 ”， 但 大 多 数 人 
都 已 经 认可 了 “博客 ”。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


博客 管理 系统 的 开发 流程 

进一步 掌握 如 何 做 项 目 需求 分 析 与 系统 设计 

实现 一 个 简单 的 公告 栏 管理 模块 

掌握 不 同 的 图 片上 传 技术 

掌握 一 种 采用 JavaScript+CSS 技术 来 实现 的 半 透 明 动 态 下 拉 荣 单 


豆 吾 于 至 至 
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8.1 开发 背景 


博客 管理 系统 给 人 们 提供 抒发 个 人 情感 、 人 与 人 之 间 进 行 良好 沟通 的 平台 , 博客 拥有 真实 的 内 容 ， 
人 们 可 以 通过 博客 记录 下 工作 、 学 习 、 生 活 和 娱乐 的 点 滴 ， 以 及 发 表 文 章 和 评论 ， 从 而 在 网 上 建立 一 
个 完全 属于 自己 的 个 人 天 地 ， 因 此 博客 成 为 当今 网 络 最 为 个 性 化 和 平民 化 的 个 人 展示 空间 。 对 于 网 民 
只 要 拥有 博客 ， 就 可 以 超越 现实 生活 ， 拥 有 不 同 以 往 的 全 新 网 上 生活 ， 对 于 博客 服务 商 ， 则 必须 从 功 
能 提供 转型 到 全 方位 社会 服务 的 提供 ， 建 立 虚拟 社会 ， 并 负责 维护 运行 ， 保 证 博客 日 常生 活 ， 对 于 社 
会 而 言 ， 有 利于 构建 和 谐 的 互联 网 空间 ， 维 护 和 谐 的 社会 环境 。 从 这 个 角度 来 说 ， 构 建新 生活 方式 ， 
将 是 互联 网 发 展 的 一 个 里 程 碑 。 


8.2 需求 分 析 


信息 时 代 的 今天 ， 博 客 已 经 成 为 一 种 新 的 生活 方式 。 在 网 络 中 构建 一 个 赋 有 个 性 化 的 个 人 博客 
提供 了 一 种 可 信任 的 和 实时 连通 的 网 络 环境 ， 通 过 网 络 开放 性 和 交互 性 的 特点 ， 让 用 户 在 任何 时 间 、 
任何 地 点 ， 通 过 网 络 方便 地 “生活 ”， 不 仅 是 信息 传递 与 获取 ， 还 可 以 进行 群体 交流 和 资源 共享 ， 展 
示 自 我 ， 为 个 人 发 展 带 来 新 机 遇 。 
通过 对 多 个 博客 网 的 调查 分 析 ， 客 户 要 求 本 博客 管理 系统 具有 以 下 功能 ; 
要 求 系统 采用 B/S 架构 ， 实 现 人 机 交互 。 
要 求 系统 界面 个 性 化 ， 色 彩 搭配 和 谐 ， 有 很 强 的 视觉 冲击 力 ， 操 作 简 便 。 
要 求 突出 主题 ， 显 示 最 新 文章 和 公告 。 
要 求 游客 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评论 。 
要 求 具 有 强大 的 搜索 查询 功能 ， 实 现 精 确 查询 和 模糊 查询 。 
完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 ， 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功能 ， 可 以 上 传 各 种 类 型 的 图 片 。 
支持 好 友 功能 。 
系统 运行 稳定 ， 安 全 可 靠 。 


办 办 办 办 办 办 多 多 多 


8.3 系统 设计 


8.3.1 系统 目标 


该 系统 主要 实现 如 下 目标 : 
回 “系统 采用 B/S 架构 ， 实 现 人 机 交互 。 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


系统 界面 设计 以 浅 色 为 主 ， 美 观 友好 ， 操 作 简便 。 

突出 重点 内 容 ， 显 示 最 新 文章 。 

非 登录 用 户 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评 论 。 

全 面 的 搜索 查询 功能 ， 包 括 精确 查询 和 模糊 查询 。 

完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功 能 。 

支持 好 友 功 能 。 

支持 公告 栏 功能 。 

系统 运行 稳定 ， 安 全 可 靠 。 


办 办 办 办 办 多 多 


8.3.2 系统 功能 结构 


博客 管理 系统 的 功能 结构 如 图 8.1 所 示 。 


删 
除 
好 
友 


图 8.1 博客 管理 系统 功能 结构 图 
8.3.3 ”系统 功能 预览 


为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 图 ， 如 果 想 查 
看 完整 的 效果 图 ， 请 参见 光盘 源 程序 。 

博客 管理 系统 的 首页 如 图 8.2 所 示 ， 该 页 面包 含 了 系统 大 部 分 的 功能 链接 ， 包 括 用 户 注册 、 用 户 
登录 、 文 章 浏览 等 。 用 户 注册 页 面 如 图 8.3 所 示 ， 该 页 面 显示 了 用 户 注册 时 需要 填写 的 资料 、 注 意 事 
项 等 。 

浏览 文章 页 面 如 图 8.4 所 示 ， 该 页 面 用 于 显示 文章 及 相关 的 评论 ， 也 可 以 发 表 评论 。 发 表 文 章 页 面 
如 图 8.5 所 示 ， 该 页 面 用 于 登录 用 户 发 表 文 章 ， 包 括 文章 标题 、 文 字 编辑 区 和 文章 内 容 。 

图 片上 传 页 面 用 于 上 传 图 片 ， 如 图 8.6 所 示 , 通过 该 页 面 用 户 可 以 将 图 片 或 照片 添加 到 博客 中 。 添 
加 好 友 页 面 如 图 8.7 所 示 ， 该 页 面 用 于 输入 用 户 好 友 的 详细 信息 ， 包 括 姓名 、 性 别 、 生 日 等 。 
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el Es 


图 8.2 博客 首页 (光盘 \TM\08\online\index.php) 图 8.3 用 户 注册 页 面 (光盘 \TM\08\online\Register.php) 
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图 8.6 图 片上 传 页 面 (光盘 \TMVWO8\onlinevadd picphp) 8.7 添加 好 友 页 面 (光盘 \TM\08\online\friendd.php) 
8.3.4 系统 流程 图 


博客 管理 系统 的 流程 图 


8.8 所 示 。 


a 
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浏 
览 
图 
片 


8.8 系统 流程 图 
8.3.5 开发 环境 


在 开发 博客 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
， 服 务 器 端 
操作 系统 : Windows Server 2003 
服务 器 : Apache 2.0。 
PHP 软件 : PHP 5.0。 
数据 库 : MySQL 5.0。 
MySQL 图 形 化 管理 软件 : PhpMyAdmin-2.5.5。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 
客户 端 
浏览 器 : 推荐 使 用 正 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 1024X768 像素 。 


8.3.6 文件 夹 组 织 结构 


国 回国 加 加 回回 加 一 


网 网 和 


博客 系统 的 目录 比较 少 ， 结 构 比 较 简 单 ， 主 要 有 数据 库 链接 文件 目录 、ess 样式 目录 、js 脚本 目录 
及 背景 图 片 目录 。 文 件 夹 组 织 结构 如 图 8.9 所 示 。 
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com 数据 库 链 接 文件 
css ss 样式 文件 

images ”图片 青 景 上 录 

x javaseript 肚 本 文件 


上 
§ 


图 8.9 ”博客 管理 系统 文件 夹 组 织 结构 
8.4 数据 库 设 计 


8.4.1 数据 库 分 析 


本 系统 属于 中 小 型 个 人 网 站 ， 毫 无 争议 的 ， 本 系统 采用 的 依然 是 PHP+MySQL 这 对 黄金 组 合 ， 无 
论 是 从 成 本 、 性 能 、 安 全 上 考虑 ， 还 是 从 易 操作 性 上 考虑 ，MySQL 都 是 最 佳 选 择 。 


8.4.2 ”数据库 概念 设计 


通过 需求 分 析 和 功能 上 的 设计 ， 本 系统 规划 出 用 户 信息 实体 、 上 传 图 片 实体 、 朋 友 圈 实体 、 文 章 
实体 和 留言 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 

用 户 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 在 本 系统 中 进行 发 表 文章 、 上 传 图 片 等 操作 ， 
则 必须 要 先进 行 注册 。 用 户 信 息 的 实体 E-R 图 如 图 8.10 所 示 。 

上 传 图 片 实体 主要 包括 上 传 图 片 的 名 称 、 上 传 图 片 用 户 和 上 传 图 片 时 间 等 。 上 传 图 片 的 实体 E-R 
图 如 图 8.11 所 示 。 


用 户 真实 姓名 ) 。 


图 8.10 用 户 信息 实体 E-R 图 图 8.11 上 传 图 片 实体 E-R 图 


8.4.3 ”数据库 物理 结构 设计 


根据 实体 E-R 图 和 本 系统 的 实际 情况 ， 需 要 创建 6 张 数据 表 ， 如 图 8.12 所 示 。 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


加 服务 器 : localhost ， 昌 数据 库 : db_tmlog 


表 类 型 整理 说 明 
th_article MyISAM gb2312_chinese_ci 。 文章 列表 
也 _filecomment MySAM 。 gb2312_chinese_ci 。 评论 列表 
th_friend MyISAM gb2312_chinese_ci 。 好 友 列表 
th_public MyISAM “gb2312_chinese_ci 。 公告 列表 
th_tpsc MyISAM ”gb2312_chinese_ci 。 图 片 列表 
th_user MySAM ”gb2312_chinese_ci 。 用 户 列表 


图 8.12 数据 表 列表 
下 面 来 具体 看 一 下 这 6 张 数据 表 的 结构 设计 。 
1. tb_user〔 用 户 列表 ) 
用 户 列表 主要 存储 用 户 的 个 人 信息 。tb_user 表 的 结构 如 图 8.13 所 示 。 
加 服务 器 : localhost ， 怕 数据 库 : db_tmlog》 国 表 :tb_user 


字段 类 型 整理 一 Null 默认 说 明 
地 int(20) 理 自动 编号 
regname varchar(20) 。 gb2312_chinese_cl 否 用 户 账号 
regrealname varchar(20) gb2312_chinese_cl 否 真实 姓名 
regpwd varchar(40) 。 gb2312_chinese_cl 否 用 户 密 码 
regbirthday date 否 。 0000-00-00 ”用 户 生日 
regemail varchar(100) gb2312_chinese_cl 否 了 -mail 
regcity varchar(100) gb2312_chinese_cl 否 所 在 城市 
regico varchar(50) 。 gb2312_chinese_cl 理 用 户头 像 
regsex varchar(4) gb2312_chinese_cl 理 用 户 性 别 
regqq varchar(40) 。 gb2312_chinese_cl 理 用 户 og 
reghomepage varchar(100) gb2312_chinese_cl 理 用 户主 页 
regsign varchar(200) gb2312_chinese_cl 否 个 人 签名 
regintroduce = mediumted gb2312_chinese_ci 否 自我 介绍 
ip varchar(20) 。 gb2312_chinese_ci 理 和 登录 IP 
fig int(1) 否 0 管理 员 


图 8.13 用 户 列表 结构 
2. tb_article (文章 列表 ) 


文章 列表 存储 的 是 用 户 发 表 过 的 文章 信息 。tb_article 表 的 结构 如 图 8.14 所 示 。 
加 服务 器 : localhost ， 图 数据 库 : db_tmlog ， 国 表 :tb_article 


字段 类 型 整理 Null 默认 说 明 
uu int(10) 理 自动 编号 
tie varchar(200) gb2312_chinese_ci 理 文章 标题 
content mediumtext gb2312_chinese_cl 要 文章 内 容 
author varchar(20) gb2312_chinese_ci 理 文章 作者 
now datetime 否 。 0000-00-00 00:00:00 。 上 传 时 间 


图 8.14 文章 列表 结构 
3. tb_filecomment (评论 列表 ) 


评论 列表 存储 的 是 用 户 对 文章 的 评论 ,包括 注册 用 户 和 游客 都 可 以 发 表 评论 。tb_filecomment 表 的 
结构 如 图 8.15 所 示 。 
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加 服务 器 :localhost 》 晶 数据 库 : db_tmlog ， 国 表 :tb_filecomment 


及 类 型 整理 是 Ma i 说 明 
加 nt) 理 自动 编号 
Teid nt 和 ) 否 0 文章 :a 
username varchar(20) gb2312_chinese cl 否 评论 用 户 
content text 9b2312_chinese cl 。 否 评论 内 容 
datetime datetime 和 理 ”0000-00-00 00:00:00 。 评论 时 间 


图 8.15 评论 列表 结构 
4. tb_tpsc (图 片 列表 ) 


图 片 列表 存储 的 是 上 传 图 片 的 信息 ， 如 图 片 名 称 、 上 传 用 户 、 上 传 时 间 等 。tb_tpsc 表 的 结构 如 
图 8.16 所 示 。 


胃 服务 器 : localhost 怕 数据 库 : db_tmlog ， 国 表 :tb_tpsc 
FR 。。 关 型 整理 尾 性 ”Nul 。 默认 说明 


0) 本 自动 摔 导 
tpme varchar(30) 9b2312_chinese_cl 理 图 片 名 称 
Me medumblob BNARY 理 二 进 制图 片 
author varchar(20) 。 9b2312_chinese_cl 理 上 传 用 户 
cj 。 dale 否 。 0000-00-00 。 上 传 时 间 


图 8.16 图 片 列表 结构 
5. tb_friend (好 友 列 表 ) 
好 友 列 表 主 要 记录 了 姓名 、 性 别 、 生 日 等 好 友 的 个 人 信息 。tb_friend 表 的 结构 如 图 8.17 所 示 。 


加 服务 器 : localhost 》 恒 数据 库 : db_tmlog ， 加 表 :tb_friend 


字 有 & 类 型 属性 Nul 默认 说 明 
地 int(4) 否 自动 丹 呈 
name varchar(50) gb2312_chinese_ci 否 好 友 姓 名 
SeX varchar(10) 。 gb2312_chinese_ci 香 好 友 性 别 
bir date 否 。 000000.00 好友 生日 
city varchar(50) gb2312_chinese_ci 香 所 在 城市 
address varchar(100) gb2312_chinese_cl 和 理 家 庭 性 址 
postcode varchar(6) gb2312_chinese_ci 理 邮政 编码 
emall varchar(50) gb2312_chinese_cl 理 sil 
tel varchar(20) 。 gb2312_chinese_cl 理 座机 号 码 
handset varchar(20) gb2312_chinese_ci 否 0 手机 号 码 
oa varchar(20) 。 gb2312_chinese_ci 否 0 好 友 00 
Username varchar(20) 。 gb2312_chinese_ci 理 用 户 昵 称 


图 8.17 好 友 列表 结构 
6. tb_public〈 公 告 列表 ) 


公告 列表 主要 记录 了 网 站 情况 、 博 客 系统 的 版 本 情况 或 是 网 站 活动 等 。 公 告 列表 的 结构 如 图 8.18 
所 示 。 


园 服务 器 : localhost ”局 数据 库 : db_tmlog ， 国 表 :tb_public 


字 有 & 委 型 整理 屋 性 Nul 至 认 说 明 
uu int(4) 要 自动 编号 
le Varchar(50) 。 gb2312_chinese_ci 理 公告 主题 
content varchar(200) gb2312_chinese_ci 和 否 公告 内 容 
pub time date 否 发 布 时 间 


图 8.18 公告 列表 结构 
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8.5 首页 设计 


8.5.1 首页 概述 


本 系统 首页 页 面 设计 简洁 ， 主 要 包括 以 下 3 部 分 内 容 : 
回 ”首部 导航 栏 : 包括 首页 链接 、 注 册 和 登录 模块 。 
左 侧 显示 区 : 包括 最 新 文章 、 最 新 图 片 和 系统 时 间 模 块 。 游 客 主要 通过 该 区 域 浏览 文章 、 浏 
览 图 片 及 发 表 评论 。 
主 显 示 区 : 为 系统 公告 栏 ， 显 示 系 统 及 网 站 的 最 新 咨询 。 
下 面 看 一 下 本 案例 中 提供 的 首页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\08\tmlog\index.php， 如 
图 8.19 所 示 。 


31 二 -要 6-7 通 
10 11 12 13 王 :5 
17 18 19 20 21 22 
24 25 26 27 28 29 
31 


图 8.19 博客 管理 系统 首页 


8.5.2 首页 技术 分 析 


在 首页 主 显示 区 ， 是 一 个 公告 栏 模 块 。 公 告 栏 主要 用 于 公布 系统 版 本 的 更 新 或 升级 情况 、 网 站 的 
最 新 活动 安排 等 信息 ， 也 可 以 链接 一 些 用 户 的 精彩 文章 。 本 系统 的 公告 栏 模 块 是 通过 <marquee> 标 签 来 


里 
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实现 的 。<marquee> 标 签 是 HTML 自 带 的 ， 也 是 初学 者 最 常用 的 公告 栏 实现 方式 。 使 用 <marquee> 标 签 
可 以 实现 文字 或 图 片 的 滚动 效果 ， 增 加 了 实用 性 的 同时 ， 也 增加 了 观赏 性 。 下 面 就 来 学 习 一 下 该 标签 
的 常用 方法 及 属性 。 


公 注意: <marquee> 标 签 是 微软 与 网 景 等 公司 私 制 的 ， 从 来 没有 被 W3C 当 作 正 式 标签 来 使 用 ， 所 以 
除了 正 浏 览 器 外 ， 有 些 属性 是 不 被 其 他 浏览 器 所 支持 的 。 在 使 用 时 一 定 要 留意 。 

1. <marquee> 标 签 的 文字 移动 属性 

<marquee> 标 签 的 特点 就 是 可 以 使 文字 或 图 片 动 起 来 ， 在 早 些 时 候 ， 这 可 是 一 项 了 不 起 的 技术 。 随 
着 W3C 标准 的 逐渐 完善 ，<marquee> 标 签 已 经 越 来 越 少 有 用 武之 地 了 ， 但 有 些 技术 是 永远 不 变 的 ， 比 
如 <marquee> 标 签 中 的 属性 名 称 和 属性 值 ， 了 解 css 样式 表 和 JavaScript 脚本 语言 的 人 会 感觉 到 非常 熟 
悉 ， 因 为 这 些 属性 名 称 和 属性 值 都 经 常 被 提 及 和 使 用 。<marquee> 标 签 用 的 文字 移动 属性 及 属性 值 如 
表 8.1 所 示 。 


表 8.1 <marquee> 标 签 常用 的 文字 移动 属性 及 说 明 


属性 名 称 | 属性 什 应 用 举例 


标签 基本 语法 ， :外 ， 还 可 以 是 
marquee 无 村 <marquee> 你 好 ，PHP</marquee> 


文字 移动 属性 ， 分 别 表示 从 右 往 左 、 从 | <marquee direction="up"> 从 下 到 上 移 
左 往 右 、 从 下 到 上 、 从 上 到 下 动 <marquee> 

文字 移动 方式 ， 分 别 表示 沿 同一 方向 不 
behavior scroll,slide,alternate | 停 滚动 、 只 滚动 一 次 、 在 两 个 边界 内 来 


direction leftrightup.down 


<marquee behavior="scroll"> 不 停 的 


循环 播放 </marquee> 


回 滚动 


村 Se - i <marquee loop=5 behavior=slide> 只 
loop 数值 1.2.3 循环 次 数 ， 不 指定 则 表示 为 无 限 循环 循环 滚动 3 次 </marquee> 
serolimount | 数值 10.20.30… | 泳 动 速度 ， 数 值 越 大 ， 速 度 越 快 Se 

了 ! ! </marquee> 


<marquee scrolldelay="500" 
Scrolldelay ”| 数值 100.200.300… | 延 时 ， 每 动 一 次 ， 停 止 的 时 间 scrollamount=100> 停 停 走 走 
</marquee> 


2. <marquee> 标 签 的 外 观 设置 
<marquee> 标 签 不 仅 可 以 设置 文字 移动 效果 , 还 能 设置 标签 的 外 观 。 外观 包括 对 齐 方式 、 标签 底 色 、 
面积 等 。<marquee> 标 签 常用 的 外 观 属性 及 说 明 如 表 8.2 所 示 。 
表 8.2 <marquee> 标 签 常用 的 外 观 设置 属性 及 说 阴 


指定 文本 的 对 齐 方式 ”| <marquee align="middle> 我 在 中 间 </marquee> 
<marquee bgcolor="#DEEBEF"> 看 到 了 吗 ? 
</marquee> 

<marquee height="15"> 不 要 ， 太 矮 了 </marquee> 
<marquee width="100"> 哇 ! 好 大 的 房子 啊 </marquee> 


top.middle.bottom 


align 
bgcolor 颜色 值 "#000000" 


数值 10.20.30… 


滚动 栏 宽度 


height 
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3. <marquee> 标 签 的 其 他 属性 
除了 以 上 的 常规 设置 ，<marquee> 标 签 还 有 两 个 特殊 的 属性 ， 即 start0 和 stop0， 它 们 可 以 配合 
JavaScript 事件 产生 很 有 意思 的 效果 。 例 如 : 


onMouseOut="this.startO": // 当 鼠标 移出 该 区 域 时 ， 开 始 滚动 
onMouseOver="this.stopO": // 当 鼠标 进入 该 区 域 时 ， 停 止 滚动 
8.5.3 首页 的 实现 过 程 


博客 管理 系统 采用 二 分 栏 结构 ， 表 单 布局 。 具 体 实现 代码 如 下 : 
例 程 01 “代码 位 置 ， 光 扒 \TM\08\tmlog\index.php 


<?php 

Session start(); // 开 启 session 支 持 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文 件 
?> 


<table width="757" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr align="right" valign= "top"> 
<td 全 卫生 向 全 7 colspan="2" a images/head.jpg"> 
-- 上 部 导航 栏 --> 


</td> 

</a> 

<t> 
<td width="236" height="501" background=" images/left.ipg"> 
吕 <!-- 左 侧 导航 栏 --> 


</td> 
<td width="521" height="501" align="center" background=" images/right.jpe"> 
0 系统 公告 区 3 
<?php 
Sp_sql = "select * from tb_public order by id desc": /从 数据 表 中 读 取 数据 
Sp_Ist = mysql_query($p_sql.$link): /执行 SQL 语 句 
?> 
一 一 一- 一 一- <Imarquee> 标 签 开始 二 一 一 > 
0 <marquee onMouseOver=this.stopO style="WIDTH: 426px: HEIGHT: 280px" onMouseOut=this.start( 
scrollamount=2 scrolldelay=7 direction=up> 
<?php 
while($p_row =mysql fetch row(Sp_rsb){ /循环 输出 公告 标题 
?> 
[2 <a href="#" onclick="wopen=open('show_pub.php?id=<?php echo 
$p_row[0]: ?>",",height=200,width=500.scollbars=no")"><?php echo Sp_row[1]; ?></a><br> 
> 
</marquee> 
<<! -- 一 -一 -- -------- -- —---- —— —-- ———- —----- 一--: ------ > 
</td> 
</t> 
</table> 
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志和 代码 贴 十 

@ <marquee> 标 签 : <marquee> 标 签 中 包含 了 start 和 stop 两 个 特殊 的 属性 ， 还 有 过度 属性 scollamount 和 延迟 属性 
scrolldelay， 而 公告 栏 的 高 度 (height) 和 宽度 ( width) 则 是 通过 css 样式 设置 的 。 

@ <?php echo $p row[1]: ?>: 循环 输出 公告 标题 ， 并 为 每 个 公告 添加 超 链接 ， 这 里 用 到 了 JavaScript 脚本 的 open() 
方法 ， 该 方法 的 作用 是 打开 一 个 新 窗口 ， 同时 可 以 对 窗口 样式 做 出 设置 。Open() 方 法 的 一 般 格式 为 open(" 链 接 的 url"， 
" 自 定义 名 称 "," 窗 口 样式 ")。 


8.6 文章 管理 模块 设计 


8.6.1 文章 管理 模块 概述 


对 一 个 博客 系统 来 说 ， 文 章 管理 是 最 基本 的 功能 ， 但 同时 也 是 最 复杂 的 一 个 功能 。 本 系统 的 文章 
管理 模块 包括 “添加 博客 文章 ”、“ 查 找 博客 文章 ”、“ 管 理 我 的 博客 ”、“ 发 表 评 论 ” “删除 文章 ” 
和 “删除 评论 ”等 6 大 功能 。 其 中 ， 普 通用 户 只 能 删除 自己 的 文章 及 发 表 对 文章 的 评论 ， 只 有 管理 员 
才 有 权 删 除 任何 一 篇 文章 及 回复 。 下 面 给 出 文章 管理 模块 的 框架 ， 如 图 8.20 所 示 。 


8.20 文章 管理 模块 框架 图 


8.6.2 文章 管理 模块 技术 分 析 


想 要 使 用 文章 管理 模块 ， 前 提 是 用 户 必 须 登录 ， 匿 名 用 户 是 无 法 访问 这 些 功能 的 ， 想 要 删除 文章 
和 评论 ， 前 提 是 当前 用 户 要 么 是 管理 员 权 限 ， 要 么 是 文章 拥有 者 ， 否 则 不 会 显示 删除 功能 的 。 这 两 方 
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面 的 控制 都 需要 session 的 配合 ， 本 节 就 来 讲 一 下 session 的 应 用 及 常见 的 问题 处 理 。 

session 的 中 文 译名 为 “会 话 ”， 是 指 用 户 从 进入 网 站 开始 ， 直 到 关闭 网 站 这 段 时 间 内 ， 所 有 网 页 
共同 使 用 的 公共 变量 的 存储 机 制 。session 比 cookie 更 有 优势 ， 如 session 是 存储 在 服务 器 端的 ， 不 易 被 
伪造 ， session 的 存储 没有 长 度 限 制 ，session 的 控制 更 容易 等 。 

PHP 中 的 session 功能 一 直 令 许 多 初学 者 望而生畏 ， 因 为 使 用 它 会 经 常 出 现 一 些 莫名 其 妙 的 错误 ， 
而 又 不 知道 如 何 去 解 决 。 其 实 ， 大 多 数 的 错误 都 是 因为 对 session 的 配置 不 了 解 导致 使 用 方法 不 正确 造 
成 的 ， 在 php.ini 中 对 session 的 配置 如 表 8.3 所 示 。 


表 8.3 session 的 常用 配置 选项 


配置 选项 说 了 明 
session.save_path = c:/temp 保存 session 变量 的 目录 ， 在 linux/UNIX 下 为 /tmp 
session.ues_cookies =1 是 否 使 用 cookie 
session.name = PHPSESSID 表示 会 话 ID 
Session auto start = 0 是 否 自动 启用 Session， 当 为 1 时 ， 在 每 页 中 就 不 必 调 用 Session satrt0 函 数 了 
设 定 cookie 送 到 浏览 器 后 的 保存 时 间 ， 单 位 为 秒 。 默 认 值 为 0， 表 示 直 到 浏览 器 
session.cookie lifetime =0 关闭 
session.cookie path=/ cookie 有 效 路 径 
Session.cookie_ domain = 有 效 域名 
session.serialize_handler= php ”| 定义 序列 化 数据 的 标识 , 本 功能 只 有 WDDX 模块 或 PHP 内 部 使 用 , 默认 值 为 PHP 
Session .gc_probability = 1 设 定 每 次 临时 文件 开始 处 理 的 处 理 概率 。 默 认 值 为 1 


session.gc_maxlifetime = 1440 ”| 设 定 保存 Session 的 临时 文件 被 清除 前 的 存活 秒 数 

决定 参照 到 客户 端的 Session 代码 是 否 要 删除 。 有 了 时 出 于 安全 或 其 他 考虑 ,会 设 定 
不 删除 。 默 认 值 为 0 

session.cache_limiter = nocache | 设 定 Session 缓冲 限制 

session.cache expire = 180 文档 有 效 期 ， 单 位 为 分 钟 

session.save_handler = files 用 于 保存 Session 变量 ， 默 认 情况 下 用 文件 


session.referer_check = 


对 于 初学 者 来 说 ，session 在 php.ini 中 是 不 需要 特意 去 改动 的 ， 因 为 安装 时 会 根据 操作 系统 自行 做 

出 适当 的 调整 。 只 对 少数 的 几 项 ， 如 session 存活 周期 (session.cookie lifetime = 0) 、 自 动 开 启 session 
(Csession.auto_start) 等 稍 加 改动 即 可 。 

PHP 主要 是 通过 会 话 (session) 处 理 函 数 来 对 session 进行 控制 和 使 用 的 。 常 用 的 处 理 函 数 如 表 8.4 

所 示 。 
表 8.4 PHP 常用 的 会 话 处 理 函 数 
函数 说 明 

开启 session 或 返回 已 经 存在 的 session 
注册 一 个 session 变量 
设 定 或 取得 当前 的 session_ id 值 
检测 指定 的 session 值 是 否 存在 。isset 不 只 可 以 检测 session， 还 可 以 检测 其 他 类 型 ， 如 
isset($_ POST[mame])、isset($_GET[mame) 等 


函数 
Session_startO: 
$_ SESSION[name]= value: 


Session_id(O) 


isset($_ SESSION[mame]) 
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续 表 
函数 函数 说 明 


session regenerate id() 


更 改 session id 的 值 
返回 或 改变 当前 session 的 name 
删除 名 为 name 的 session。 注 意 
结束 当前 会 话 ， 删 除 所 有 session 


session name() 
unset($_SESSION['name']) 
Session destro: 
A 注意 : (1) 如 果 要 改变 当前 session 的 name 值 ， 必 须 在 session() 之 前 调用 session name() 函 数 ， 
而 且 session name 不 能 全 部 是 数字 ， 否 则 会 不 停 地 生成 新 的 session id。 
(2 ) 不 可 以 写成 unset($_SESSION)， 这 样 会 禁止 整个 会 话 的 功能 。 

在 本 节 最 后 ， 介 绍 几 个 使 用 session 时 要 注意 的 问题 。 

1. 尽 可 能 地 将 session_start 放 到 第 1 行 

这 种 情况 是 新 手 最 容易 犯 的 错误 。 产 生 的 错误 代码 为 : 

Warning: session start() [function.session-start]: Cannot send session cache limiter - headers already sent… 

其 原因 就 是 在 使 用 session_start0 之 前 ， 就 有 HTML 代码 输出 了 。 也 许 有 的 读者 会 说 : “没有 ， 
session_start 之 前 没有 任何 代码 ， 绝 对 没有 。” 那 么 ， 请 检查 你 的 程序 ， 是 不 是 有 空 行 ， 或 类 似 echo 
语句 的 输出 。 如 果 有 ， 请 去 掉 ， 因 为 就 算是 一 个 小 小 的 空格 都 是 不 可 以 的 。 所 以 ， 为 了 避免 这 类 错误 
的 发 生 ， 尽 可 能 地 将 session_start 放 到 第 1 行 。 

2. 在 使 用 session 之 前 一 定 要 先 写 session_start() 

大 多 数 读者 在 使 用 session 之 前 都 能 先 调用 session_start0 函 数 , 但 对 于 session_destroy0 函 数 却 经 常 
忽略 。session_destroy0 虽 然 是 结束 当前 会 话 并 删除 所 有 session， 但 在 删除 之 前 ， 也 要 先 开 启 session 支 
持 才 可 以 ， 不 然 会 产生 这 样 的 错误 代码 : 

session_destroy() [function.session-destroy]: Trying to destroy uninitialized session in… 

所 以 ， 凡 是 在 使 用 session 或 session 函数 的 页 面 中 ， 都 要 加 上 session_start0 这 句 话 。 

3. 删除 所 有 session 

如 果 想 删除 所 有 session, 但 又 不 想 结束 当前 会 话 ， 用 unset 一 个 一 个 删除 实在 是 太 麻烦 了 ,最 简单 
的 办 法 就 是 将 一 个 空 数组 赋 给 $_SESSION， 如 $_SESSION = array0， 这 样 就 解决 了 。 


8.6.3 ”添加 文章 的 实现 过 程 


国 ”添加 文章 模块 使 用 的 数据 表 : tb_article 

当 用 户 登录 后 ， 系 统 会 直接 进入 到 文章 添加 页 (file php) ， 也 可 以 通过 选择 “文章 管理 ”/“ 添 加 
博客 文章 ”菜单 项 回 到 file php 页 。 文 章 添加 页 面 的 运行 结果 如 图 8.21 所 示 。 

添加 文章 页 为 一 个 发 布 表单 ， 包 括 文章 主题 、 文 字 编辑 、 文 章 内 容 等 元 素 。 部 分 表单 元 素 如 表 8.5 
所 示 。 
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二 是 ;| 厅 天 中 竺 要 换 诺 了 7” ] 
WE: 了 | 了 | 了 | 了 人 [FF Fs [Be [ARE 


《font sire=4 eolor= 和 000099) 中 午 吃 百 芽 我 不 萤 吃 凋 。555555555555555555</font》 当 
文章 内 容 : 
| 夯 | 
图 8.21 添加 文章 页 面 的 运行 结果 
表 8.5 添加 文章 页 面 的 主要 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含义 
myform form method="post" action="check file.php" 添加 文章 表单 
txt_title text id="txt_title" size="68" 文章 标题 
nt el name="font" Sloe ne oe 文章 字体 
onChange="showfont(this.options[this.selectedIndex].value)" 
lass="wenbenkuang" 
size select ee 上 字体 大 小 
onChange="showsize(this.options[this.selectedIndex].value)” 
nChange="showenlor(this.optionsl thaselecteamdery vetoo) 字体 颜色 
name="color" size="1" class="wenbenkuang" id="select" 
file textarea cols="75" rows="20" id="file" style="border:0px:width:520px:" 文章 内 容 
btn 1 submit id="btn_ tj" value=" 提 交 " onClick="return check0:" “提交 ”按钮 


当 用 户 填 写 完 博客 主题 和 文章 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 系 统 将 跳 转 到 处 理 页 (check file.php) 
进行 处 理 。 在 处 理 页 中 ， 将 传 过 来 的 文章 标题 、 文 章 作者 和 文章 内 容 等 参数 组 成 insert 语句 ， 并 最 终 保 
存 到 数据 表 中 。 如 果 添 加 信息 成 功 ， 系 统 返回 到 本 页 ， 可 继续 执行 添加 操作 ;如 果 添 加 失败 ， 则 返回 
到 上 一 步 。 程 序 的 关键 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 VTMWg8\tmlog\check file php 


<2php 

session_start(): // 开 启 session 支 持 

include "Conn/conn.php": // 包 含 数据 库 链 接 文 件 
if($Sbtn 1<—""){ // 判 断 传 值 页 面 
Stitle=$_POST[txt_title]: /取得 文章 标题 

$author=$_ SESSION[usemame]: /取得 作者 

$content=$_ POST[file]: // 取 得 文章 内 容 
$now=date("Y-m-d H:i:s"): /使 用 date 函 数 生成 发 布 时 间 
诺 生成 insert 语 句 */ 


$sql="Insert Into tb_article (title.content.author.now) Values (‘$title'.'$content'.'$author'.'$now")": 
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Sresult=mysql query($sq]): /执行 insert 语 句 
上 # 根据 Sresult， 返 回 结果 */ 
这 Sresulb{ 
echo "<script>alert(' 巷 喜 您 ， 你 的 文章 发 表 成 功 !11"):window.location.href='file.php':</script>": 
} 


else{ 
echo "<script>alert( 对 不 起 ， 添 加 操作 失败 !11"):history.go(-1):</script>"; 
. 
}else{ 
echo "<script>alert( 对 不 起 ， 添 加 操作 失败 !!11"):history.go(-1):</script>"; 
1 
2> 
人 说 明 : 文章 添加 页 面 中 ， 使 用 了 部 分 UBB 语法 ， 由 于 UBB 语法 不 属于 本 书 的 范畴 ， 所 以 这 里 不 
做 讲解 ， 请 感 兴趣 的 朋友 查看 相关 的 书籍 。 


8.6.4 文章 列表 的 实现 过 程 
国 ”查看 文章 列表 使 用 的 数据 表 : tb_article 


选择 “文章 管理 ”/“ 我 的 文章 ”菜单 项 , 将 显示 用 户 发 表 过 的 文章 列表 。 文章 列表 页 面 (myfiles.php) 
的 运行 结果 如 图 8.22 所 示 。 


查看 我 的 文章 
| 1 今天 中 午 要 控 贱 了 ~ 
页 次 : 1/1 页 记录 : 1 条 


8.22 文章 列表 页 面 的 运行 结果 
文章 列表 页 面 使 用 了 分 页 技术 和 do…while 循环 语句 来 输出 文章 标题 。 程 序 的 关键 代码 如 下 : 


例 程 03 ”代码 位 置 : 光盘 \TM\08\tmlog\myfiles.php 

<?php 

Session _start(): /开启 session 支 持 
include "Conn/conn .php": /包含 数据 库 链 接 文件 
include "check login.php": // 包 含 权限 检查 文件 
?> 

Fi We 


<2php 
启 ”Spage 为 当前 页 ， 如 果 Spage 为 空 ， 则 初始 化 为 1 */ 
if (Spage—""){ 
Spage=1:} 
© if(is numeric(Spage)){ 
Spage_size=20; // 每 页 显示 20 条 记录 
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Squery="select count(*) as total from tb _article where author = ".$_SESSION[username]." order by id desc": 


Sresult=mysql_query($query): // 查 询 符 合 条 件 的 记录 总 条 数 
$message_count=mysql_result($result.0."total"): // 要 显示 的 总 记录 数 
Spage_count=ceil($message_count/$page size): // 根 据 记 录 总 数 除 以 每 页 显示 的 记录 数 求 出 所 分 的 页 数 
Soffset=($page-1)*$page_size: // 计 算 下 一 页 从 第 几 条 数据 开始 循环 


S$sql=mysql_query("select id,title from tb _article where author = ".$_ SESSION[username]." order by id desc limit 


Soffset, $page_size"): 


Sinfo=mysql fetch array($sq)): 
?> 


<2php 
必 输出 结果 集 */ 
这 Sinfo){ 
3i=1; /文章 序号 
do{ //do…while 循 环 开始 
?> 
<t> 
<!-- 显示 文章 序号 和 文章 标题 --> 
<td width="498" align="left” valign="top"> &nbsp;é&nbsp;&nbsp:<a href="showmy.php?file id=<?php echo 


S$info[id]:?>"><?php echo Si."、".Sinfo[title]:?></a> </td> 


</t> 

<?php 

$i=$i+1: 

}while(Sinfo=mysql_fetch_array(SsqD) // 循 环 结束 
?> 

</table></td> 

</t> 

</table></td> 

</t> 

<?php } ?> 


<!--” 翻 页 条 --> 
<td width="33%">&nbsp;&nbsp: 页 次 : <?php echo Spage:?>/<?php echo Spage_count:?> 页 &nbsp: 记 录 : <?php echo 


Smessage_count:?> 条 &nbsp: </td> 
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<td width="67%" align="right" class="hongse01"> 

<?php 

族 ”如果 当 前 页 不 是 首页 */ 

if($page!=1){ 

全 显示 “首页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=1> 首 页 </a>&nbsp:"; 

少 显示 “上 一 页 ” 超 链接 */ 

echo "<a href~myfiles.php?page=".($page-1)."> 上 一 页 </a>&nbsp:": 


} 

谨 ” 如 果 当 前 页 不 是 尾 页 */ 

这 $page<$page_count) 

沁 

必 显示 “下 一 页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=".($paget+1)."> 下 一 页 </a>&nbsp:"; 
放 显示 “ 尾 页 ” 超 链接 六 
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echo "<a href-myfiles php?page=".$page_count."> 尾 页 </a>": 
沙 
?> 
Ah 代码 贴 二 
@ is numeric(Spage): is_numericO 函 数 判 断 变量 $page 是 否 为 数字 ， 如 果 是 数字 ， 则 返回 true， 否 者 返回 false。 如 
果 一 组 字符 全 部 由 数字 组 成 ， 那 么 i numeric(0) 函 数 也 将 返回 true。 
@ do…while0 循 环 和 while0 循 环 : do…while() 循 环 是 先 执 行人 f} 中 的 代码 段 ， 然 后 判断 while 中 的 条 件 表达 式 是 否 成 
立 ， 如 果 成 立 返回 ture， 则 重复 输出 {} 中 的 内 容 ， 否 则 结束 循环 ， 执 行 while 下 面 的 语句 。While 循环 是 先 判 断 while 中 
的 表达 式 ， 当 返回 ture 时 ， 再 执行 人 中 的 代码 。 两 者 的 区 别 是 : do…while0 循 环比 while0 循 环 多 输出 1 次 结果 。 


8.6.5 查看 文章 、 评 论 的 实现 过 程 


国 ”查看 文章 、 评 论 使 用 的 数据 表 : tb_article、tb_filecomment 
单 击 列表 中 任意 一 个 文章 标题 , 都 会 看 到 对 应 的 文章 内 容 和 文章 评论 。 查 看 文章 页 面 (showmy.php) 
的 运行 结果 如 图 8.23 所 示 。 


博客 ID 号 55 作者 spon 发 表 时 间 2007 2119 
全 攻 主 题 今天 中 和 要 控 绩 了 ~ 

中 午 吃 豆芽 ， 我 不 受 吃 啊 pcm 
~55555555555555555555555 


评论 人 q3939 评论 时 间 2007-12-19 13:24:59 
评论 内 容 这 就是 偏食 的 结果 ,HOHO~~ ^0 人 X 尖 出 除 


_ 豆 :7 珊 亨 :1 条 


8.23 查看 文章 页 面 的 运行 结果 


系统 根据 当前 页 面 传 过 来 的 文章 id 值 从 数据 表 tb_article 中 返回 对 应 的 文章 信息 (包括 文章 id、 文 
章 作者 、 文 章 标题 、 文 章 内容 和 发 表 时 间 ) 、 输 出 文章 信息 后 ， 开 始 查 找 表 tb_filecomment 中 fileid 字 
段 值 等 于 文章 id 的 所 有 评论 集 ， 并 通过 分 页 显示 出 来 。 显 示 文 章 页 面 (showmy.php) 的 关键 代码 如 下 : 
例 程 04 ”代码 位 置 : 光盘 \TM\08\tmlog\showmy.php 
全 


<2php 
session start():; // 开 启 session 支 持 
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include "Conn/conn .php": /包含 数据 库 链 接 文件 
include "check login php": // 包 含 权限 检查 文件 
Sfile idl=$_ GET[file id]: /取得 文章 ID 号 
$bool = false: // 判 断 用户 删 除权 限 , 默认 的 false 是 没有 权限 
De 
<!-- 首部 导航 栏 -> 
Se es Se 


<?php 
$sql=mysql_query("select * from tb_article where id = ".$file id1): /根据 文章 ID 号 查找 文章 
Sresult=mysql_fetch_array($sq]l): // 返 回 结果 集 


二 二 星 示 文 章 二 二 一 二 堆 
00%" border="1" cellpadding="1" cellspacing="1" bordercolor="#D6E7AS" bgcolor="#FFFFFF" 
class="i table"> 
<tr bgcolor="#FFFFFF"> 
<td width="14%" align="center"> 博 客 ID 号 </td> 
<1-- 2 显示 文章 jd------------------------------ 
<td widi 5%"><?php echo Sresult[id]: ?></td> 
<td width="11%" align="center"> 作 者 </td> 
<!-- ---- 
<td width="18%"><?php echo Sresult[author]; ?></td> 
<td width="12%" align="center"> 发 表 时 间 </td> 
<!-- -一 显示 发 表 时 间 - 一 -一 -一 > 
<td widi 0%"><?php echo Sresult[now]: ?></td> 
</tr> <tr bgcolor="#FFFFFF"> 
<td align="center"> 博 客 主题 </td> 
到- 文章 主题 一 一 一 一 一 一 一 一 一 一 一 一 > 
<td colspan="S">&nbsp:&nbsp:<?php echo Sresult[title]: ?></td> 
</tr> <tr bgcolor="#FFFFFF"> 
<td align="center"> 文 章 内 容 </td> 


<1-- -一 ------- 

<td colspan="4"><?php echo Sresult[content]: ?></td> 

<td> 
<2php 

局 ”判断 登录 用 户 是 否 为 管理 员 ， 或 者 是 文章 的 作者 */ 

这 $_SESSION[fig] 一 1 or ($_ SESSION[usemame] 一 Sresult[author])){ 
证 如 果 是 ， 就 将 Sbool 设 为 true */ 
$bool = 

?> 

二 同时 显示 “删除 ”按钮 ， 并 将 文章 id 作 为 url 后 级 一 起 传 到 处 理 页 ----------------- 二 


<a href="del file.php?file id=<?php echo $result[id]:?>"><img src="images/A_delete.gif' width="52" height="16" alt=" 
删除 博客 文章 " onClick="return d_chk0:"></a> 

<?php } ?> 

<?php 

we // 分 页 代码 部 分 略 
$sql=mysql_query("select * from tb_filecomment where fileid='$file id1' order by id desc limit $offset $page_ size"): 
Sresult=mysql_fetch_array($sql): 
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/本 村 本 于 于 本 疝 训 证 可 字 字 让 本 本 字 本 本 本 本村 二村 本本 本本 二 本 本 二 本 本 本 根据 返回 记录 集 输 出 记录 于 让 本 本 本 本 相机 本 本 机 机 本 本 本 下 本 本事 可 机 市 可 可 本 可 本事 束 字 机 束 字 */ 
if($result—false){ 
echo "<font color=#ff0000> 对 不 起 ， 没 有 相关 评论 !</font>": 
}else{ 
dof 
?> 
<t> 
<td height="57" align="center" valign="top" ><table width="480" border="1" cellpadding="1" 
cellspacine="1" bordercolor="#D6E7AS" bgcolor="#FFFFFF" class="i table"> 
<tr bgcolor="#FFFFFF"> 
<td width="14%" align="center"> 评 论 ID 号 </td> 
< 上 -一 一 一 -一 评论 id -> 
<td width="15%"><?php echo Sresult[id]; ?></td> 
<td width="11%" align="center"> 评 论 人 </td> 
< -一 = 评论 大 一 一 一 一 = 二 
<td width="18%"><?php echo Sresult[username]: ?></td> 
<td width="12%" align="center"> 评 论 时 间 </td> 
< 社论 时 间 二 = == 
<td width="30%"><?php echo Sresult[datetime]: ?></td> 
</t> 
<tr bgcolor="#FFFFFF"> 
<td align="center"> 评 论 内 容 </td> 
和 论 内 站 Ty 
4"><?php echo Sresult[content]: ?></td> 


<?php 

族 如 果 S$bool 为 真 ， 输 出 删除 超 链 接 ， 否 则 跳 过 */ 
if ($booD{ 
?> 


<a href="del_comment.php?comment id=<?php echo $result[id]?>"><img src="images/A_delete.gif' width="52" 
height="16" alt=" 删 除 博客 文章 评论 " onClick="return d_chk0:"></a> 
<?php }?> 
</td></tr> </table></td> </t> 
<?php 
}while($result=mysql_fetch_array($sqD)); 

} 
?> 
</table> 
Ss <!--” 翻 页 功能 代码 部 分 略 -> 


8.6.6 ”删除 文章 、 评 论 的 实现 过 程 


国 ”查看 文章 、 评 论 模块 使 用 的 数据 表 : tb_article、tb_filecomment 

在 查看 文章 评论 页 面 ， 当 系统 判定 当前 用 户 为 管理 员 或 是 文章 作者 时 ， 在 每 篇 文章 和 评论 的 后 面 ， 
都 将 显示 相应 的 “删除 ”按钮 。 单 击 任意 的 “删除 ”按钮 ， 系 统 会 提示 是 否 删除 ， 如 果 确认 ， 将 跳 转 
到 处 理 页 (del file.php 和 del commentphp) ， 完 成 删除 操作 。 

在 删除 文章 的 处 理 页 中 ， 删 除 文章 的 同时 ， 也 删除 了 该 篇 文章 的 相关 的 评论 。 处 理 页 首先 在 文章 
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列表 (tb_article) 中 删除 id 等 于 $file id 的 记录 ， 如 果 没 有 可 删除 记录 ， 则 提示 失败 ， 并 返回 上 一 步 ; 
如 果 删 除 成 功 ， 则 转 到 评论 列表 〈tb_filecomment) 中 ， 删 除 所 有 该 篇 文章 的 评论 。 删 除 文章 页 
(del filephp) 的 关键 代码 如 下 : 

例 程 05 “代码 位 置 : 光盘 \TMWOSmlogvdel file php 


<2?php 
session start(); // 开 启 session 支 持 
include "check login php": // 包 含 权限 检查 文件 
include "Conn/conn .php": /包含 数 据 库 链接 文件 
$sql="delete from tb_article where id=".$file_id; // 删 除 文章 的 delete 语 句 
Sresult=mysql_query($sql.$link): // 执 行 删 除 操作 
这 $resulb{ 
$sqll = "delete from tb_filecomment where fileid = ".Sfile id; /删除 相对 应 的 文章 评论 
Srstl = mysql_ query($sqll,Slinlk: /执行 删除 评论 的 操作 
ifsrstD) 
echo "<script>alert(' 博 客 文章 已 被 删除 :location= myfiles.php]':</script>": 
else 


echo "<script>alert( 删 除 失败 1"):history.go(-1):</script>"; 
| 


else{ 
echo "<script>alert(' 博 客 文 章 删除 操作 失败 !"):history.go(-1):</script>"; 


?> 


因为 删除 文章 的 过 程 也 包含 了 删除 评论 的 过 程 ， 所 以 这 里 就 不 给 出 删除 评论 的 代码 了 。 
8.7 图 片上 传 模块 设计 


由 于 动态 网 络 编程 技术 的 诞生 ， 使 得 网 络 更 加 人 性 化 ， 为 了 能 够 和 用 户 更 好 地 互动 ， 很 多 网 站 都 
提供 了 让 用 户 上 传 图 片 的 功能 。 一 个 网 站 拥有 图 片上 传 功能 是 非常 必要 的 。 
8.7.1 图 片上 传 模块 概述 

图 片上 传 在 动态 网 页 开发 过 程 中 应 用 非常 广泛 。 如 果 有 比较 好 的 图 片 想 和 其 他 人 一 同 分 享 ， 就 可 
以 通过 图 片上 传 功能 来 实现 ， 以 增加 网 站 的 核心 竞争 力 。 本 系统 的 图 片上 传 模块 主要 实现 对 图 片 的 添 


加 、 浏 览 、 查 询 和 删除 操作 ， 而 对 图 片 的 删除 则 只 有 管理 员 才 有 权限 。 图 片上 传 模块 框架 图 如 图 8.24 
所 示 。 


8.7.2 图 片上 传 模块 技术 分 析 


既然 是 图 片上 传 模块 ， 显 而 易 见 本 节 的 主要 技术 就 是 上 传 图 片 功能 了 。 上 传 图 片 和 上 传 文件 的 原 
理 基 本 相同 ， 下 面 就 来 一 步 一 步 地 学 习 如 何 上 传 图 片 和 图 片 的 两 种 保存 方式 。 
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普通 用 户 超级 用 户 


图 8.24 图 片上 传 模块 框架 图 
1. 上 传 图 片 的 基本 流程 


在 网 页 中 实现 上 传 图 片 功能 的 步 又 如 下 : 
(1) 通过 <form> 表 单 中 的 file 元 素 选取 上 传 数 据 。 


使 用 file 元素 上 传 数据 时 注意 一 点 : 就 是 在 form 表单 中 要 加 上 属性 enctype="multipart/form-data"， 
否则 上 传 不 了 文件 〈 图 片 ) 。 


(2) 在 处 理 页 中 使 用 $_FILES 变量 中 的 属性 判断 上 传 文件 类 型 和 上 传 文件 〈 图 片 ) 大 小 是 否 符合 


$_FILES 变量 为 系统 预定 义 变量 ， 保 存 的 是 上 传 文件 〈 图 片 ) 的 相关 属性 。 使 用 格式 为 
$_FILES[name][property]: 


图 片 的 相关 属性 如 表 8.6 所 示 。 
表 8.6 $_FILES 的 相关 属性 
属 性 值 说 了 明 
name 上 传 文件 的 文件 名 
i 上 传 文件 的 类 型 
Size 上 传 文件 的 大 小 
tmp name 上 传 文件 在 服务 器 中 的 临时 文件 名 
elITOT 上 传 文件 失败 的 错误 代码 


(3) 使 用 move_ uploaded file0 函数 上 传 文件 〈 图 片 ) 或 将 文件 〈 图 片 ) 以 二 进 制 的 形式 保存 到 
数据 库 中 。 


使 用 函数 将 文件 (图片) 保存 到 对 应 的 文件 夹 中 和 以 二 进 制 的 形式 保存 到 数据 库 中 是 上 传 文件 (图 
片 ) 的 两 种 形式 ， 稍 后 将 单独 做 介绍 。 
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(4) 返回 页 面 等 待 下 一 步 操作 。 

2. 使 用 上 传 函数 保存 文件 〈 图 片 ) 

使 用 上 传 函数 上 传 文件 〈 图 片 ) 的 本 质 就 是 将 文件 〈 图 片 ) 从 浏览 器 端 复制 到 服务 器 端 指定 的 文 
件 夹 里 ， 数 据 库 所 存储 的 就 是 文件 〈 图 片 ) 的 相对 地 址 。 当 页 面 显示 图 片 时 ， 实 际 是 分 两 步 : 第 1 步 
是 读 取 数 据 表 中 的 地 址 ;第 2 步 是 根据 地 址 找到 并 在 页 面 中 显示 图 片 。 使 用 目录 保存 文件 的 好 处 是 减 
少 了 数据 库 的 容量 和 对 数据 库 的 压力 ， 而 且 图 片 很 容易 被 搜索 引擎 抓 到 ， 从 而 提高 网 站 流量 和 人 气 。 

move_uploaded_file() 函 数 的 一 般 格式 为 : 

bool move_uploaded file ( string filename, string destination ): 

回 flename: 上 传 到 服务 器 中 的 临时 文件 名 。 

回 destination: 保存 文件 的 实际 路 径 。 


名 注意 : 这 里 的 filename 为 临时 文件 名 ， 而 不 是 上 传 文件 的 原文 件 名 ， 可 以 通过 $_FILES[filename] 
[tmp_name] 来 获取 。 
下 面 看 一 个 实例 ， 程 序 代码 如 下 : 


<label> 请 选择 要 上 传 的 图 片 〈 文 件 小 于 $00K) : </label> 
<!-- 上 传 文件 的 form 表 单 --> 


<input type="hidden" name="action" value="upload" /> 
<input type="file" name="u file"/> 
<input type="submit" value=" 上 传 " /> 

</form> 


<?php 
谨 判断 是 否 为 上 传动 作 */ 
ifisset($_ POST[action])){ 


$file_path = "./": /上 传 文件 存放 路 径 ，./ 为 当前 目录 下 
Sprofix = array("jpg".".gif' "jpeg"): /设置 允许 上 传 的 文件 后 缀 类 型 
$f name =$_FILES[u file][name]: // 取 得 要 上 传 的 文件 名 
S$pro_name=substr($f_name,strrpos($f name.".")): // 取 得 上 传 文件 的 后 级 
if(!in_array($pro_name, $profix){ // 判 断 上 传 文件 的 类 型 是 否 为 允许 类 型 
echo "文件 格式 不 对 ": 
exitO: 
} 
if($_FILES[u file][size] <= 0){ // 判 断 上 传 文件 的 大 小 


echo "文件 上 传 错误 ， 或 文件 大 于 500K. 请 重新 上 传 "””// 如 果 文 件 过 大 ， 提 示 错 误 
} 


else{ 
人 上 传 文件 函数 */ 
Imove_Uploaded file($ FILES[u file][tmp namel].$file path.$ FILES[u_file][name]): 
echo "上 传 成 功 !": 
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3. 使 用 二 进 制 保存 图 片 

上 传 图 片 的 另 一 种 保存 方式 是 以 二 进 制 的 形式 存储 在 数据 库 中 。 在 计算 机 看 来 ， 再 美的 图 片 、 再 
感人 的 文章 ， 也 不 过 就 是 一 堆 “0111010011…” 的 代码 段 而 已 ， 和 其 他 数据 没有 什么 区 别 ， 存 到 数据 
表 中 都 是 一 样 的 。 存 储 图 片 不 再 需要 使 用 move_uploads_file0) 函 数 ,在 处 理 页 中 直接 使 用 二 进 制 的 形式 
读 取出 文件 图 片 ， 并 存 到 数据 表 中 。 本 系统 采用 的 就 是 二 进 制 的 形式 来 保存 图 片 ， 具 体操 作 将 在 实现 
过 程 里 给 出 。 

使 用 二 进 制 来 保存 图 片 的 好 处 是 安全 ， 特 别 是 涉及 到 个 人 隐私 ， 不 易 被 窃取 ; 存储 方便 ， 和 表 中 
其 他 数据 同等 操作 就 可 以 ;节省 磁盘 空间 ， 由 于 文件 系统 类 型 的 限制 ， 放 到 数据 表 中 的 数据 一 定 比 直 
接 放 到 磁盘 中 要 节省 空间 。 

当然 二 进 制 形式 的 图 片 也 有 很 大 次 端 ， 在 传 出 的 过 程 中 ， 如 果 某 个 二 进 制 位 出 现 丢 失 或 损坏 ， 那 
么 整 张 图 片 将 无 法 显示 ， 而 是 变 为 一 堆 乱 码 ， 而 且 如 果 存 储 的 图 片 过 多 ， 会 造成 数据 库 过 于 庞大 ， 从 
而 影响 其 他 数据 的 正常 读 取 ， 增 大 服务 器 的 负担 。 所 以 ， 对 于 太 大 的 文件 ， 就 不 适宜 保存 到 数据 库 中 
而 是 应 该 放 到 文件 夹 。 


8.7.3 图 片上 传 的 实现 过 程 


国 ”查看 文章 、 评 论 模块 使 用 的 数据 表 : tb tpsc 

博客 用 户 登录 后 ， 选 择 导航 栏 中 的 “图 片 管理 ”/“ 添 加 图 片 ”菜单 项 ， 即 可 进入 添加 图 片 页 面 ， 
在 “图 片 名 称 ”文本 框 中 添加 上 传 的 图 片 名 称 ， 在 “上 传 路 径 ” 文 本 框 中 选择 或 者 单 击 “ 浏 览 ” 按 钮 
选择 自己 喜欢 的 图 片 ， 单 击 “ 提 交 ” 按 钮 ， 以 二 进 制 形式 将 图 片上 传 到 数据 库 中 。 图 片上 传 页 面 的 运 
行 结果 如 图 8.25 所 示 。 


8.25 图 片上 传 页 面 的 运行 结果 


图 片上 传 页 是 一 个 上 传 文件 的 表单 ， 主 要 包括 一 个 文本 域 、 一 个 文件 域 和 一 个 “提交 ”按钮 。 部 
分 表单 元 素 的 名 称 及 属性 如 表 8.7 所 示 。 


表 8.7 图 片上 传 页 面 中 的 表单 元 素 


method="post" action="tpt]_ok.php" _ enctype="multipart/form-data" 
type="text" id="tpmce" size="40" 
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续 表 


重要 属性 含 
type="file" size="23" maxlength="60" | 上 传 路 径 
type="submit" id="btn_tj" value=" 提 交 " onClick="return pic chkO:" “提交 ”按钮 
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file file 


submit 


当 用 户 输入 图 片 名 称 , 并 选择 图 片 路 径 后 , 单 击 “提交 ”按钮 , 系统 将 进入 到 上 传 处 理 页 (tptj_ok.php) 
中 进行 处 理 。 在 处 理 页 中 ， 首 先 对 图 片 名 称 进行 处 理 ， 去 掉 特殊 字符 、 空 行 和 空格 ， 然 后 对 上 传 的 文 
件 进 行 类 型 检查 、 文 件 大 小 检查 。 最 后 以 二 进 制 的 形式 ， 和 图 片 的 其 他 信息 (如 上 传 用 户 、 上 传 时 间 
等 ) 一 起 存 进 数 据 表 中 。 关 键 代 码 如 下 : 


例 程 06 ”代码 位 置 : 光盘 \TM\08\tmlog\tptj_ok.php 


<?php 
session start(): // 开 启 session 支 持 
include "check login php": // 包 含 权限 检查 文件 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文 件 
这 $_POST["btm_t"] 一 "提交 "){ 
@ Stpmc=htmlspecialchars($tpmc): // 将 图 片 名 称 中 的 特殊 字符 转换 成 HTMI 格 式 
2 Stpme=str_replace("\n","<br>",$tpme): // 将 图 片 名 称 中 的 回 车 符 以 自动 换行 符 取 代 
S$tpmc=str_replace("","é&nbsp:",$tpme); // 将 图 片 名 称 中 的 空格 以 "&nbsp;" 取 代 
$author=$_ SESSION[username]: 
S$scsj=date("Y-m-d"): 1/ 设置 图 片 的 上 传 时 间 
Sprofix = array(".jpe",".gif",".jpeg"): // 设 置 允 许 上 传 的 文件 后 级 类 型 
$f name =$_FILES[file][name]: // 取 得 要 上 传 的 文件 名 


S$pro_name=substr($f_name,strrpos($f_ name,".")); ”// 取 得 上 传 文件 的 后 级 
谨 判断 上 传 文件 的 类 型 是 否 为 允许 类 型 */ 
© if(!in_array(strtolower($pro_name), Sprofix)){ 
echo "<script>alert(' 文 件 格式 不 对 "):history.go(-1):</script>"; 
exit(); 


上 
让 判断 上 传 文件 的 大 小 ， 如 果 文 件 过 大 ， 提 示 错 误 */ 
if($_FILES[file][size] > 500000){ 

echo "<script>alert(' 文 件 上 传 错误 ,请 重新 上 传 ):history.go(-1)</script>"; 


exitO; 
} 
else{ 
Sfp=fopen($file."r"): // 以 只 读 方 式 打开 文件 
Sfile=addslashes(fread($fp.filesize($file))): // 将 文件 中 的 引号 部 分 加 上 反 斜 线 


$query="insert into tb_tpsc (tpmc.,file,author,scsj) values ('$tpmce','$file','$author','$scsj")": 
Sresult=mysql_query($query): 
echo "<meta http-equiv=\"refresh\" content=\"1:url=browse_pic.php\"> 图 片上 传 成 功 ， 请 稍 等 ..…"; 


} 


?> 


< 代码 贴 二 
@ htmlspecialchars0 函 数 : 将 特殊 字符 转换 成 HTML 格式 。 
@ str replace0 函 数 : 取代 所 有 在 字 串 中 出 现 的 字 串 ， 语 法 如 下 : 
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mixed str_ Teplace ( mixed search. mixed replace. mixed subject . int &count ) 


str_Ieplace() 函 数 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ,参数 &count 表 示 替 换 字 符 串 执行 的 次 数 。 
上 @ strtolower: 将 字符 转换 为 小 写字 母 。 


8.7.4 图 片 浏览 的 实现 过 程 


国 图片 上传 使 用 的 数据 表 : tb tpsc 

无 论 是 注册 用 户 ， 还 是 非 注册 用 户 ， 只 要 登录 网 站 ， 就 可 以 无 条 件 地 浏览 所 有 图 片 。 而 删除 图 片 
除了 管理 员 ， 其 他 人 都 无 权 操作 。 非 注册 用 户 可 以 通过 首页 中 的 “最 新 图 片 ” 进 入 图 片 浏览 页 面 ， 注 
册 用 户 先 进入 个 人 管理 界面 ， 选 择 “ 图 片 管理 ”/“ 浏 览 图 片 ” 菜 单项 ， 同 样 可 以 进入 图 片 浏览 页 面 。 
注册 用 户 浏览 图 片 页 面 的 运行 结果 ， 如 图 8.26 所 示 。 


1 2007-12- 


:007-12-14 


页 次 :1/1 页 记录 :2 条 
图 8.26 浏览 图 片 页 面 的 运行 结果 
本 页 的 实现 代码 和 查看 文章 页 面 略 有 不 同 ， 在 查看 文章 页 面 中 ， 每 条 数据 占 了 一 行 ， 而 查看 图 片 


则 采用 的 是 分 栏 显 示 ， 以 每 行 两 张 图 片 的 格式 输出 ， 每 页 显示 4 张 图 片 。 通 过 单 击 小 图 片 ， 可 以 查看 
图 片 的 原 效果 图 。 程 序 的 关键 代码 如 下 : 


例 程 07 代码 位 置 ， 光盘 \TM\08\tmlog\browse_pic.php 


<2php 
session start(): // 开 启 session 支 持 
include "Conn/conn php": // 包 含 数据 库 链 接 文 件 
include "check login.php": // 包 含 权限 检查 文件 
?> 


四 <!-- 首 部 导航 栏 -> 
<tr align="left" colspan="2" > 
<td width="390" height="25" colspan="3" valign="top" bgcolor="#EFF7DE"> <span class="tableBorder LTR"> 浏 
览 图片 </span></td></tr> 
<t> 
<td height="192" align="center" valign="top" > 
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<2php 
» // 分 页 显示 
Squery="select * from tb_tpsc where scsj order by id desc limit $offset. Spage size": 
请 返回 结果 集 */ 
S$result-mysql query($query): 
> 
<table width="496" border="1" align="center" cellpadding="3" cellspacine="1" bordercolor="#D6D7D6"> 
<tr> 
<?php 
/* ”设置 一 个 变量 ,判断 当前 数据 是 奇数 输出 还 是 偶数 输出 */ 
Si=1; 
while($info=mysql fetch array($result)) 
{ 
/* ”如 果 当 前 为 偶数 输出 ， 记 录 输 出 完毕 后 结束 <tr> 标 签 、*/ 
if($i%2—0){ 
?> 


<td width="500"><table width="245" border="0" cellpadding="0" cellspacing= 
<tr> 
<td colspan="2"><div align="center"> 
<!-- 显 示 图 片 页 image.php --> 
<a href="image.php?recid=<?php echo $info[id]; ?>" target="_blank"><img 
sic="image.php?recid=<?php echo $info[id]:?>" width="150" height="100"></a></div></td></t> 
<t> 
<td width="109" height="25" align="left">&nbsp; 图 片 名 称 :<?php echo 


Sinfo[tpmc]:?></td> 
<td width="128"> 上 传 时 间 :<?php echo $info[scsj]:?></td></t> 
<tr> 
<td colspan="2" height="25"> 
<2php 
上 * “如果 登 录用 户 为 管理 员 ， 显 示 删 除 操作 。*/ 
if($_SESSION[fig]—1){ 
?> 
<a href="remove.php?pic_id=<?php echo $info[id]?>"><img 
src="images/A_delete.gif" width="52" height="16" alt=" 删 片 " onClick="return pic_chkO:"></a> 


<2php 
} 
> 
<td> 
</t> 
</table></td> 
</tr> 
<2php 
上 
睛 否 者 ， 开 始 新 的 一 行 */ 
else 
{ 
> 
<td width="500" ><table width="236" border="0" cellpadding="0" cellspacine="0"> 
<t> 


<td colspan="2"><div align="center"><a hre 全 "image.php?recid=<2php echo $info[id]: ?>" 
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target="_blank"><img src="image.php?recid=<?php echo $info[id]:?>" width="150" height="100"></a></div></td> 
</u> 
<t> 
<td width="110" height="25">&nbsp: 图 片 名 称 :<?php echo Sinfo[tpmc]:?></td> 
<td width="126"> 上 传 时 间 :&nbsp:<?php echo Sinfo[scsj]:?></td> 
</t> 
<t> 
<td colspan="2" height="25"> 
<?php 
这 ($S_ SESSION[fig] 一 TD){ 
?> 
<a href="remove.php?pic id=<?php echo $info[id]?>"><img 
src="images/A_delete.gif" width="52" height="16" alt=" 删 除 图 片 " onClick="returmn pic_chkQ:"></a> 


<?php 
| 
?> 
</td> 
</t> 
</table></td> 


<?php 


; 
上 # 变量 $i 自 +1，while 循 环 结束 */ 
Si++; 


} 


> 
</tr> 


通过 上 述 代 码 可 以 看 到 ， 预 览 图 片 和 显示 图 片 实际 调用 的 都 是 同一 页 面 image.php。image.php 页 
就 是 用 来 显示 图 片 的 实际 效果 。 实 现代 码 如 下 : 
例 程 08 ”代码 位 置 : 光盘 \TM\08\tmlog\image.php 


<?php 
include "Conn/conn .php": /包含 数据 库 链 接 文件 
S$query="select id.tpmc,file.author.scsj from tb_tpsc where id=".$recid; 。“ // 根 据 id 生 成 查询 语句 
Sresult=mysql_query($query): // 执 行 查询 语句 
if(!S$result) die("error: mysql query"): // 判 断 是 否 有 返回 结果 
Snum=mysql_num rows(Sresult): // 计 算 结果 集 个 数 
if($num<]1) die("error: no this recorder"): 
S$data = mysql_result(Sresult.0,"file"): // 返 回 图 片 数据 
echo $data: // 输 出 图 片 

?> 


名 注意 : mysql_resultO 函 数 不 能 和 其 他 处 理 结果 集 函数 混合 调用 。 
8.7.5 删除 图 片 的 实现 过 程 


国 。 删除 图 片 使 用 的 数据 表 : tb tpsc 
删除 图 片 是 管理 员 才 有 的 权限 ， 在 图 片 浏览 的 实现 代码 中 ， 已 经 给 出 如 何 判断 登录 用 户 是 否 为 管 
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理 员 。 当 管理 员 单 击 “ 删 除 ” 超 链接 时 ， 处 理 页 (removephp) 会 根据 传 过 来 的 id 删除 对 应 的 数据 表 
中 的 数据 。 实 现代 码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\08\tmlog\remove.php 


<?php 
session start(): // 开 启 session 支 持 
include "check login php": // 包 含 权限 检查 文件 
include "Conn/conn php": // 包 含 数据 库 链接 文件 
这 $_session[fig] != "1"){ // 判 断 是 否 为 管理 员 
echo "<script>alert( 您 没有 删除 权限 "):history.go(-1):</script>"; 
S$sql="delete from tb_tpsc where id=".$pic id: // 生 成 删除 语句 
Sresult=mysql query($sqD: /执行 删除 语句 
if($result){ // 根 据 $result 返 回 结 果 
echo "<script>alert( 图 片 删除 成 功 !"");location='browse_pic.php':</script>"; 
else{ 
echo "<script>alert( 图 片 删除 操作 失败 !"):history.go(-1):</script>"; 
} 
?> 


8.7.6 单元 测试 


在 进行 上 传 模块 的 测试 时 发 现 ， 当 上 传 比较 大 的 图 片 时 ， 文 件 大 小 的 验证 部 分 并 没有 生效 ， 而 是 
直接 跳 过 。 在 处 理 页 中 产生 错误 ， 并 且 显 示 页 面 不 显示 图 片 。 运 行 结果 如 图 8.27 所 示 。 
Warning: fread(): supplied argument is not a valid stream resource 
in D:NAppServNwwwNtmlog\tptj_ok-php on line 27 


图 片上 传 成 功 ， 请 稍 等 . . . 
| 图 片 


图 上 名 称 :sdfsdaf 二 全 条 ;32007123 尘 片 名 称 : 查 名 的 水 上 传 时 间 :2007-12-20 


X 删除 XX 删除 
页 次 ; 1/2 页 记录 : 5 条 下 一 页 尾 页 
图 8.27 上 传 图 片 错 误 信 息 及 显示 结果 
查看 原文 件 大 小 验证 的 代码 如 下 : 


谨 判断 上 传 文件 的 大 小 ， 如 果 文 件 过 大 ， 提 示 错 误 */ 
if($_FILES[file][size] > 500000){ 
echo "<script>alert( 文 件 上 传 错 误 ， 请 重新 上 传 ):history.go(-1)</script>": 
exitO: 
} 


经 反复 测试 后 发 现 : 如 果 上 传 图 片 大 于 500KB 而 小 于 2MB 时 ， 原 代码 生效 。 如 果 图 片 大 于 2MB， 
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则 代码 被 跳 过 ， 这 时 输出 $_FILES[file][size] 的 值 为 0。 原 来 ， 在 配置 文件 php.ini 中 ， 
upload mix filesize=2MB， 也 就 是 说 ， 上 传 文件 的 大 小 超过 了 系统 设置 的 值 ， 那 么 $_FILES[file][size] 
返回 0， 在 判断 表达 式 中 返回 true， 代 码 自然 被 跳 过 。 

改写 验证 代码 ， 更 改 后 的 代码 如 下 : 

例 程 10 ”代码 位 置 : 光 玲 \TM\08\tmlog\tptj_ok -php 


if(($_FILES[file][size] <= 0) or ($_FILES[file][size] > 500000){ 
echo "<script>alert( 文 件 上 传 错 误 ， 或 文件 大 于 500K， 请 重新 上 传 ):history.go(-1)</script>"; 


exitO; 


} 
进行 二 次 测试 ， 问 题 没 有 再 出 现 ， 系 统 正常 运行 。 


8.8 ”朋友 圈 模 块 设计 


8.8.1 朋友 圈 模 块 概述 


本 系统 的 朋友 圈 模 块 的 主要 功能 是 添加 、 查 询 、 删 除 好友 ， 添 加 的 好 友 除 了 该 用 户 以 外 ， 包 括 管 
理 员 在 内 的 所 有 外 人 都 不 可 以 查看 ， 以 保证 其 个 人 隐私 不 被 外 泄 。 用 户 被 删除 时 ， 该 用 户 现 有 的 朋友 
圈 也 一 并 被 删除 。 朋 友 圈 模块 框架 图 如 图 8.28 所 示 。 


好 友 列表 


图 8.28 朋友 圈 模 块 框架 图 


8.8.2 ”朋友 圈 模 块 技术 分 析 


在 查询 好 友 的 功能 中 ， 使 用 到 了 模糊 查询 语句 ， 用 于 模糊 查找 好 友 列 表 。 模 糊 查询 语句 使 用 的 是 
like 运算 符 。 在 PHP 中 ， 带 有 like 运算 符 的 查询 语句 的 常用 格式 有 2 种 : 
(1) 使 用 通配符 “%” 的 where 子 句 
通配符 “%” 表 示 0 或 多 个 任意 长 度 和 类 型 的 字符 ， 包 括 中 文 汉字 。 
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示例 1: 表示 查找 所 有 内 容 包 含 “ 好 ” 字 的 文章 。 
select * from tb_file where content like % 好 %': 
示例 2: 查找 所 有 包含 “好 ” 字 或 “高 ” 字 的 文章 ， 这 时 可 以 配合 or 运算 符 来 使 用 ， 代 码 如 下 : 
select * from tb_file where content like '% 好 %' or content like '% 高 %'; 
(2) 使 用 通配符 “_” 的 where 子 句 
通配符 “_” 表 示 匹 配 任意 的 单个 字符 。 
示例 1: 查找 用 户 名 只 包含 5 个 字符 ， 其 中 后 4 个 字符 为 soft 的 用 户 ， 代 码 如 下 : 
select * from tb_user Where regname like ' soft; 
示例 2: 查找 所 有 以 + 开头 、 并 且 以 + 结尾 的 、 中 间 包 含 3 个 字符 的 用 户 ， 代 码 如 下 : 
select * from tb_user Where regname like tt’: 


查找 的 结果 为 tsoft。 


全 注意; 使 用 MySQL 做 模糊 查询 要 注意 编码 问题 。 如 果 编码 不 统一 ， 那 么 查询 时 就 容易 查 不 到 数 
据 ， 或 返回 的 数据 不 匹配 。 所 以 在 安装 MySQL 时 ， 要 保持 和 系统 编码 的 统一 。 常 用 的 编码 
格式 包括 gb2312、ISO-8859-1、utf8 和 gbk 等 。 


8.8.3 ”查询 好 友 的 实现 过 程 


国 ”查看 文章 、 评 论 模块 使 用 的 数据 表 : tb_friend 

当 用 户 要 查询 好 友 时 ， 选 择 “朋友 圈 管 理 ”/“ 查 询 朋 友信 息 ” 菜 单项 ， 显 示 查 询 页 面 。 查 询 可 以 
分 为 姓名 查询 和 编号 查询 ， 均 为 模糊 查询 。 当 用 户 输入 要 查找 的 关键 字 后 ， 单 击 “ 检 索 ” 按 钮 ， 或 按 
Enter 键 ， 系 统 跳 到 处 理 页 进行 处 理 。 查 询 好 友 页 面 的 运行 结果 如 图 8.29 所 示 。 


柄 号 28 姓名 tsoft 性 别 0 生日 。 1980-01-14 
所 在 城市 ”上 海 柠 明 案 隆 性 址 训 林 
邮政 辆 码 13**** 宗 庭 电话 “0431-12345ere 


e-mail 。 zh#### 久 163.com 


手机 号 码 “1351XXxx359 QQ 1l234eme 关 开除 


8.29 查询 好 友 页 面 的 运行 结果 
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查询 页 包含 一 个 查询 表单 ， 包 括 查 询 条 件 和 查询 关键 字 两 部 分 表单 元 素 。 主 要 表单 元 素 如 表 8.8 
所 示 。 


表 8.8 查询 页 表单 的 主要 元 素 属 性 


元 素 类 型 重要 属性 含义 
form | method="post" action="query_friend.php" onSubmit="return checkO:” | 查询 好 友 表 单 
<option value="name" selected> 姓 名 </option> 


et <option value="id"> 编 号 </option> 查询 杀 件 寺 择 
查询 关键 字 


“检索 ”按钮 


myform 


sel 1t] 


text | id="sel key" size="30" 
type="submit" name="submit" value=" 检 索 " 


sel key 


submit 


当 处 理 页 接收 到 查询 条 件 及 查询 关键 字 后 ， 生 成 模糊 查询 语句 ， 执 行 SQL 语句 并 返回 查询 结果 。 
如 果 没 有 输入 关键 字 , 则 弹出 提示 框 ; 如 果 没 有 查找 到 任何 结果 , 则 输出 “Sorry! 没 有 您 要 找 的 朋友 !”。 
处 理 页 的 关键 代码 如 下 : 

例 程 11 代码 位 置 : 光盘 \TM\08\tmlog\query_friend.php 


<?php 

session start(): /开启 session 支 持 
include "Conn/conn.php"; /包含 数据 库 链 接 文件 
include "check login.php": /包含 权限 检查 文件 
?> 


<script language="javascript"> 
让 ”JavaScript 肢 本， 验证 输入 信息 ， 如 果 为 空 ， 则 弹出 提示 框 。*/ 
function check(form){ 
if (document.myform.sel key.value—""){ 
alert(" 请 输入 查询 条 件 !"); 


myform.sel_ key.focusO: 


Tetum false; 
} 
} 
</script> 
<?php 
if($_POST[sel key]!=""){ 
S$tji=$_POST[sel 1]; /接收 查询 字段 
S$key=$_POST[sel key]: /接收 查询 关键 字 
$sql=mysql_query("select * from tb_friend where $t like '%$key%' and username="$_SESSION[usemame]"): 
Sresult=mysql_fetch_array($sq)): // 执 行 查询 语句 
这 Sresult 一 false){ 
echo ("[<font color=red>Sorry! 没 有 您 要 找 的 朋友 !</font>]"): 
jelse{ 
?> 
议 <!-- 显示 记录 表单 -> 
<2php }2> 
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8.9 ”开发 技巧 与 难点 分 析 


想象 一 下 动态 的 下 拉 菜 单 、 半 透明 的 背景 效果 ， 鼠 标 划 过 时 的 色彩 交替， 给 人 一 种 亦 真 亦 幻 的 视 
觉 冲 击 和 美的 感受 ， 这 样 的 网 站 具有 更 大 的 吸引 力 。 本 系统 个 人 管理 页 面 中 的 管理 菜单 实现 的 就 是 这 
种 效果 。 当 用 户 登 录 到 博客 网 站 后 ， 将 鼠标 移动 到 “朋友 圈 管 理 ” 等 导航 链接 上 时 ， 将 在 下 方 显示 出 
半 透 明 的 下 拉 式 菜单 ， 透 过 此 下 拉 菜 单 仍 可 以 看 到 页 面 上 的 内 容 ， 运 行 效果 如 图 8.30 所 示 。 


查看 我 的 文章 
1、 今 天 中 个 要 按 红 了 ~ 


页 次 : 1/1 页 记录 ; 1 条 


8.30 动态 半 透 明 效果 的 下 拉 菜 单 


实现 半 透 明 背 景 的 下 拉 菜 单 ， 首 先 需要 在 页 面 中 实现 下 拉 菜 单 ， 然 后 再 通过 设置 下 拉 菜 单 的 CSS 
样式 实现 半 透 明 效果 。 实 现下 拉 菜 单 的 半 透 明 效 果 可 以 应 用 CSS 样式 的 透明 效果 滤 镜 alpha 实现 。alpha 
属性 是 把 一 个 目标 元 素 与 背景 混合 。 这 种 “与 背景 混合 ”通俗 地 说 是 一 个 元 素 的 透明 度 ， 透 明 效果 滤 
镜 alpha 的 语法 如 下 : 

{filter:alpha(opacity=opacity,finishopacity=finishopacity.style=style,startx=startx. starty=starty,finishx=finishx ,finishy=fin 
ishy)} 

滤 镜 alpha 的 各 个 参数 说 明 如 表 8.9 所 示 。 


表 8.9 alpha 滤 镜 的 各 个 参数 及 说 明 
属 性 说 阴 
代表 透明 度 水 准 。 默认 的 范围 是 0~100， 其 实 是 百分比 的 形式 ,也 就 是 0 代表 完全 透明 ，100 代表 完 
名 四 于 全 不 透明 
finishopacity | 可 选 ， 如 果 想 要 设置 渐变 的 透明 效果 ， 可 以 使 用 该 参数 指定 结束 时 的 透明 度 。 范 围 也 是 0~100 


Style 指定 透明 区 域 的 形状 特征 ， 其 中 0 代表 统一 形状 、1 代表 线形 、2 代表 放射 状 、3 代表 长 方形 
Startx 代表 渐变 透明 效果 的 开始 坐标 

Starty 代表 渐变 透明 效果 的 开始 立 坐标 

Finishx 代表 渐变 透明 效果 结束 和 坐标 

Finishy 代表 渐变 透明 效果 结束 并 坐标 


在 本 实例 中 的 menuskin 类 中 ， 设 置 了 alpha 滤 镜 的 属性 ， 代 码 如 下 : 
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例 程 12 ”代码 位 置 : 光盘 \TM\08\tmlog\css\style.css 


.menuskin { 
BORDER: #666666 1px solid: // 层 边框 样式 
VISIBILITY: hidden: // 层 可 见 为 隐藏 
FONT: 12px Verdana: // 层 中 的 字体 样式 和 大 小 
POSITION: absolute: // 定 位 方式 
background-image:url("../images/item out.gif"): /背景 图 片 
background-repeat : Tepeat-y: // 图 片 是 否 可 重复 
Filter: Alpha(Opacity=85): // 设 置 alphpa 的 opacity 属 性 等 于 85， 即 透明 度 为 85% 


} 


在 显示 页 面 中 显示 菜单 项 的 表单 引入 这 个 menuskin 类 ， 当 鼠标 划 过 特定 的 文字 时 ， 将 会 发 现 自动 
下 拉 菜 单 的 效果 为 半 透 明 的 样式 。 页 面 关键 代码 如 下 : 

例 程 13 代码 位 置 : 光盘 \TM\08\tmlog\file.php 

<!-- 弹出 层 设置 --> 

<div class=menuskin id=popmenu 
<!-- 鼠标 进入 该 区 域 时 调用 的 js 方法 “--> 
onmouseover="clearhidemenu():highlightmenu(event,'on")" 
<!-- 鼠标 离开 该 区 域 时 调用 的 js 方法 --> 
onmouseout="highlightmenu(event,'off );dynamichide(event)" 
<!-- ”设置 层 全 顺序 和 定位 方式 --> 
style="Z-index:100:;position:absolute:"> 

</div> 


<!-- 半 透 明 的 动态 下 拉 菜 单 ”--> 
<TABLE style="WIDTH: 580px" VERTICAL-ALIGN: text-top: cellSpacing=0 cellPadding=0 align="center"> 
<TR align="center" valign="middle"> 
<!-- 显示 登录 用 户 -> 
<TD style="WIDTH: 100px: COLOR: red:"> 欢 迎 您 :&nbsp:<?php echo 
$_SESSION[username]: ?>&nbsp:&nbsp:</TD> 
<!-- ”显示 首页 超 链接 _--> 
<TD style="WIDTH: 80px: COLOR: red:"><SPAN style="FONT-SIZE: 9pt COLOR: #cc0033"> 
</SPAN><a href="index.php"> 博 客 首 页 </a></TD> 
<!-- ”显示 文章 管理 下 拉 菜 单 -> 
<TD style="WIDTH: 80px: COLOR: red:"><a onmouseover=showmenu(event.productmenu) 
‘onmouseout=delayhidemenu() class='navlink’ style="CURSOR:hand" > 文章 管理 </a></TD> 
<!-- ”显示 图 片 管理 下 拉 菜 单 ”--> 
<ID style="WIDTH: 80px: COLOR: red:"><a onmouseover=showmenu(evenLHonourmenu) 
onmouseout=delayhidemenu() class='mavlink style="CURSOR:hand"> 图 片 管理 </a></TD> 
<!-- ”显示 朋友 图 管理 下 拉 菜 单 ”--> 
<ID style="WIDTH: 90px: COLOR: red:"><a onmouseover=showmenu(event.myfriend) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 朋友 图 管理 </a></TD> 
<?php 
刻 ”如 果 登 录用 户 为 管理 员 */ 
if($_SESSION[fig]—1){ 


= -显示 管理 员 管理 下 拉 莱 单一- 三 
<TID style="WIDTH: 80px: COLOR: red:"><a onmouseover=showmenu(event.myuser) 
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onmouseout=delayhidemenu0) class=mavlink' style="CURSOR:hand" > 管理 员 管理 </a></TD> 


<2php 
?> 
< -一 一 一 一 显示 退出 超 链接 -一 一 一- 一 -一 一 -一 一 -一 一 > 
<TD style="WIDTH: 80px: COLOR: red:"><a href="safe.php"> 退 出 登录 </a></TD> 
</TR> 
</TABLE> 


通过 代码 可 以 发 现 ， 在 每 个 超 链接 标签 <a> 里 ， 都 有 3 个 参数 ， 分 别 为 : 

onMouseOver: 鼠标 进入 该 区 域 时 的 效果 。 本 页 面 调 用 了 js 方法 showmenu()。 

加 ”onMouseOut: 鼠标 移出 该 区 域 时 的 效果 。 本 页 面 调用 js 方法 delayhidemenu()。 

回 style: 标签 样式 ， 本 页 面 样式 为 "CURSOR:hand:"， 显 示 一 个 手 的 形状 。 

调用 的 js 方法 就 是 实现 下 拉 菜单 的 JavaScript 代码 : 显示 和 隐藏 菜单 。 由 于 这 段 代 码 是 通用 型 代 
码 ， 所 以 单独 放 到 一 个 js 文件 中 ， 方 便 其 他 页 调用 ，js 文件 名 为 menu.js。 调 用 时 在 页 面 中 加 入 如 下 代 
码 即 可 。 


<script src="JS/menu.JS"></script> 


menu.js 文件 中 显示 、 隐 藏 菜单 和 变换 背景 等 关键 函数 的 代码 如 下 : 


例 程 14 ”代码 位 置 : 光盘 \TM\08\tmlog\js\menu.js 
启 ”浏览 器 版 本 号 检测 */ 
Var IE4=document.all&&navigator.userAgent.indexOf("Opera")=—-1 
Var netscape6=document.getElementById&&!document.all 
Var netscape4=document.layers 
/ 履 显示 下 拉 菜 单 函数 */ 
function showmenu(e,vmenu,mod){ 
请 判断 浏览 器 版 本 */ 
if (!document.all&&!document.getElementById&&!document.layers) 
return 

让 菜单 内 容 */ 

which=vmenu 

必 ”调用 clearhidemenu0 函 数 */ 

clearhidemenu() 

广 ”调用 IE_clearshadow0 函 数 */ 

IE clearshadow() 

谨 ”根据 不 同 的 浏览 器 ， 声 明 弹出 式 菜单 */ 

menuobj=IE4? document.all.popmenu : netscape6? document.getElementById("popmenu") : netscape4? 
document.popmenu : "" 

menuobj.thestyle=(IE4|lnetscape6)? menuobj.style : menuobj 

证 GE4|netscapeG) 

请 如 果 是 IE4 或 netscape6 浏 览 器 ， 就 调用 innerHTMIL 方 法 输出 下 拉 菜 单 */ 

menuobj.innerHTML=which 

else{ 

族 ” 否 者 使 用 write 方 法 输出 下 拉 菜 单 */ 
menuobj.document.write('<layer name="other" bgColor="#E6E6E6" width="165" 

‘onmouseover="clearhidemenu()" onmouseout="hidemenu()">'+which+'</layer>') 


卢 关闭 数据 流 */ 
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menuobj.document.close() 


} 
睛 ”调用 不 同 的 方法 属性 ， 来 设置 不 同 的 浏览 器 的 宽 、 高 、 及 内 部 表格 的 大 小 、 位 置 */ 
menuobj.contentwidth=(IE4|Inetscape6)? menuobj.offsetWidth : menuobj.document.other.document.width 
menuobj.contentheight=(IE4|Inetscape6)? menuobj.offsetHeight : menuobj.document.other.document.height 
eventX=IE4? event.clientX : netscape6? e.clientX : e.x 
eventY=IE4? event.clientY : netscape6? e.clientY : ey 
var rightedge=IE4? document.body.clientWidth-eventX : window.innerWidth-eventX 
Var bottomedge=IE4? documentbody.clientHeight-eventY : window.innerHeight-eventY 
if (rightedge<menuobj.contentwidth) 
menuobj.thestyle.left=IE4? document.body.scrollLeft+eventX-menuobj.contentwidth+menuOffX 
netscape6? window.pageXOffsetteventX-menuobj.contentwidth : eventX-menuobj.contentwidth 
else 
menuobj.thestyle.left=IE4? IE_x(event.srcElement)+menuOffX : netscape6? window.pageXOffset+eventX : 
eventX 
让 (bottomedge<menuobj.contentheight&&mod!=0) 
menuobi.thestyle.top=IE4? 
document.body.scrollTop+eventY-menuobj.contentheight-event.offsetY+menuOffY-23 : netscape6? 
window.pageYOffsetteventY-menuobj.contentheight-10 : eventY-menuobj.contentheight 
else 
menuobj.thestyle.top=IE4? IE_y(event.srcElement)+menuOffYy : netscape6? 
window.pageYOffsetteventY+10 : eventY 
让 ”设置 标签 样式 为 可 见 */ 
Imenuobj.thestyle.visibility="visible" 
心 ”调用 IE_dropshadow( 方 法 */ 
IE dropshadow(menuobj."#999999".3) 


return false 
} 
儿 隐藏 下 拉 菜 单 函数 */ 
function hidemenuO{ 
if (window.menuob]j) 
menuobj.thestyle.visibility=(IE4|Inetscape6)? "hidden" : "hide" 
IE clearshadow() 
} 
function dynamichide(e){ 
if (IE4&&!menuobj.contains(e.toElement)) 
hidemenu() 
else if (netscape6&&e.currentTarget!= e.relatedTarget&& !contains netscape6(e.currentTarget, e.relatedTarget)) 
hidemenu() 


} 
族 ”延迟 隐藏 下 拉 菜 单 函数 */ 
function delayhidemenu(){ 
if (IE4lInetscape6|Inetscape4) 
delayhide=setTimeout("hidemenu(O)".500) 


} 
局 ”停止 隐藏 菜单 函数 */ 
function clearhidemenuO){ 
if (window.delayhide) 
clearTimeout(delayhide) 
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} 


局” 设置 菜单 背景 */ 

店 ”鼠标 移入 该 区 域 时 显示 的 背景 */ 
function overbg(tdbe){ 
tdbg.style.background='url( images/item over.gif)' 
tdbg.style.border=' #9CA6C6 1px solid' 


j 

必 ”鼠标 移出 该 区 域 时 显示 的 背景 */ 

function outbg(tdbg){ 

tdbg.style.backeround="url( images/item out.gif)' 

tdbg.style.border=" 

} 

局 下 拉 菜 单 内 容 */ 

作文 章 管理 下 拉 菜 单 */ 

Var productmenu='<table width=90><tr><td id=fileadd onMouseOver=overbg(fileadd) onMouseOut=outbg(fileadd)><a 
href=file.php> 添 加 博客 文章 </a></td></tr>\ 

<tr><td id=query onMouseOver=overbg(query) onMouseOut=outbg(query)><a href=query.php> 查 询 博 客 文章 
</a></td></tr>\ 

<tr><td id=myfiles onMouseOver=overbg(myfiles) onMouseOut=outbg(myfiles)><a href=myfiles.php> 我 的 文章 
</a></td></tr></table>' 

作 图 片 管理 下 拉 菜 单 */ 

Var Honourmenu='<table width=90><tr><td id=picadd onMouseOver=overbg(picadd) onMouseOut=outbg(picadd)><a 
href=add_pic.php> 添 加 图 片 </a></td></tr>\ 

<tr><td id=browse onMouseOver=overbg(browse) onMouseOut=outbg(browse)><a href=browse_pic.php> 浏 览 图 片 
</a></td></tr>\ 

<tr><td id=querypic onMouseOver=overbg(querypic) onMouseOut=outbg(querypic)><a href=query_pic.php> 查 询 图 片 
</a></td></tr></table>' 

履 朋友 轿 管 理 下 拉 菜 单 */ 

Var myfriend='<table width=90><tr><td id=friendadd onMouseOver=overbg(friendadd) 
onMouseOut=outbg(friendadd)><a href=friend.php> 添 加 到 朋友 圈 </a></td></tr>\ 

<tr><td id=browse_fri onMouseOver=overbg(browse_fri) onMouseOut=outbg(browse_fri)><a hre 人 =browse_friphp> 浏 览 
我 的 朋友 </a></td></tr>\ 

<tr><td id=cxfriend onMouseOver=overbg(cxfriend) onMouseOut=outbg(cxfriend)><a href=query_friend.php> 查 询 朋 友 
信息 </a></td></tr></table>' 

上 # 管理 员 管理 下 拉 菜 单 */ 

Var myuseI='<table width=90><tr><td id=queryuser onMouseOver=overbg(queryuser) 
onMouseOut=outbg(queryuser)><a href=queryuser.php> 查 询 用 户 信 息 </a></td></tr>\ 

<tr><td id=browseuser onMouseOver=overbg(browseuser) onMouseOut=outbg(browseuser)><a href=browseuser.php> 浏 
览 用 户 信息 </a></td></tr>\ 

<tr><td id=managepub onMouseOver=overbg(managepub) onMouseOut=outbg(managepub)><a href=managepub.php> 
公告 管理 </a></td></tr\ 

</table>" 


8.10 ”登录 验证 码 技术 专题 


当今 网 络 安全 越 来 越 受 到 人 们 的 


视 , 开发 一 个 带 验证 码 的 用 户 登 录 模 块 可 以 提高 网 站 的 安全 性 。 
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其 中 ， 验 证 码 是 随机 生成 的 ， 可 以 是 字母 、 数 字 或 汉字 ， 也 可 以 是 图 片 。 本 节 技 术 专 题 主要 就 是 学 习 
一 下 验证 码 的 几 种 实现 方法 及 技巧 。 


8.10.1 简单 的 数字 验证 


先 来 看 一 个 简单 的 数字 验证 码 ， 生 成 一 组 4 位 的 随机 数字 ， 每 刷新 一 次 ， 显 示 的 结果 都 不 相同 。 
如 果 验 证 码 输 入 错误 ， 将 弹出 提示 窗口 。 数 字 验 证 码 的 运行 结果 如 图 8.31 所 示 。 


图 8.31 简单 的 数字 验证 运行 结果 


简单 的 数字 验证 码 在 开发 过 程 中 主要 应 用 两 个 函数 : 

回 mt rand0 函 数 

mt_rand0 函 数 主要 用 于 从 指定 参数 中 随机 去 一 个 数值 ， 语 法 如 下 : 

int mt_rand([int min], [int max]): 

例如 : 从 52 一 79 之 间 取 一 个 随机 数值 ， 代 码 如 下 : 

mt_rand(52,79) 

intval0 函 数 

intval0 函 数 主 要 用 于 将 变量 转 成 整数 类 型 ， 语 法 如 下 : 

int intval(mixed var, int [base]): 

intval0 函 数 可 将 变量 转 成 整数 类 型 。 可 省 略 的 参数 base 是 转换 的 基数 ， 默 认 值 为 10。 转 换 的 变量 
var 可 以 为 数组 或 类 之 外 的 任何 类 型 变量 。 


8.10.2 ”数字 图 形 验证 码 


数字 图 形 验证 码 相对 于 数字 验证 码 而 言 ,要 稍 复 
杂 一 些 。 数 字 图 形 验证 码 主要 应 用 mt_rand 函数 来 初 
始 化 一 组 4 位 的 随机 数 ,然后 利用 for 循环 语句 将 随 
机 生成 的 4 位 随机 验证 码 利用 数字 图 形 进行 输出 。 
图 形 验证 码 的 运行 结果 如 图 8.32 所 示 。 i 
实现 数字 图 形 验证 码 的 关键 代码 如 下 : 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


<!-- js 脚本 ， 判 断 文本 框 中 输入 的 验证 码 是 否 匹 配 --> 
<script language="javascript"> 
function check(form){ 
if(form.txt_yan.value—""){ 
alert(" 请 输入 验证 码 "):form.txt_yan.focus(:retum false: 
} 
if(form.txt_yan.value!=form.txt_hyan.value){ 
alert(" 对 不 起 ， 您 输入 的 验证 码 不 正确 !"):form.txt_yan.focus():return false:; 


} 


</script> 


<!-- 验证 表单 -> 
<form name="form" method="post" action="" onSubmit="return check(form):"> 
<!-- ”验证 码 文本 框 。--> 
<input type="text" name="txt_yan"> 
<!-- ”生成 随机 数字 图 形 验 证 码 ”--> 
<2php 
<!-- ”生成 随机 4 位 数 ”--> 

S$num=intval(mt rand(1000.9999)): 
<!-- 使 用 for 循 环 ， 输 出 随机 4 位 数 的 图 片 格式 --> 

for(GSi=0:$i<4:$i++){ 

echo "<img src=images/checkcode/".substr(strval($num).$i.1).".gif>": 


}2> 
<!-- 将 随机 数 存 入 隐藏 域 --> 
<input type="hidden" name="txt_hyan" id="txt_hyan" value="<?php echo $num:;?>"> 
<br> <br> 
<input type="submit" name="Submit" value=" 验 证 "> &nbsp: <input type="reset" name="Submit2" value=" 重 置 "> 
<!-- ”处 理 代 码 段 --> 
<?php 
必 判断 验证 码 录 入 框 是 否 为 空 */ 

if($_POST[txt_yan]!="") 
必 当 输 入 的 验证 码 和 隐藏 域 中 的 验证 码 相等 时 ， 输 出 欢迎 信息 六 

这 $_POST[txt_ yan] ==$_POST[txt hyan){ 
echo "您 输入 的 验证 码 通过 ， 感 谢 您 的 加 盟 .…"; 


3 
2 


8.10.3 ”汉字 图 形 验证 码 


汉字 图 形 验 证 码 是 目前 网 络 上 比较 流行 的 验证 码 方式 ， 这 种 验证 码 的 实现 有 一 定 的 难度 。 首 先 将 
图 片 所 对 应 的 汉字 依次 存储 到 数组 中 。 然 后 利用 Rand 函数 生成 一 个 小 于 10 的 数 ， 用 于 得 到 随机 字符 
的 位 置 ， 近 而 得 出 随机 字符 ， 并 设置 要 显示 的 随机 图 片 。 由 于 产生 的 随机 字符 串 是 4 位 的 ， 所 以 需要 
使 用 for 循环 语句 ， 根 据 数组 得 出 相应 图 片 的 路 径 ， 再 将 图 片 转换 成 数组 中 的 文字 ， 最 后 输出 到 浏览 
器 中 。 

以 汉字 图 形 化 显示 验证 码 的 完整 代码 如 下 : 

<?php 


400 
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Sstr=anay(" 大 "更 "他 "天 ", 吓 I" 客 "1" 情 " 技 "立新 中 
$word=strlen($str); 
for($i=0;$1<4:$i++){ 
Snum=rand(0,$word*2-1); 
Simg=$img."<img src=' images/checkcode/".$num.".gif width="16' height="16>": /显示 随机 图 片 
Spic=S$pic.$str[Snum]: // 将 图 片 转换 成 数组 中 的 文字 


8.11 本 章 总 结 


本 章 的 博客 管理 系统 首先 介绍 了 博客 的 基本 概念 、 发 展 前 景 、 影 响 范围 及 博客 网 的 功能 分 类 ， 使 
读者 对 当今 主流 博客 有 了 一 个 大 致 的 认识 。 其 次 ， 实 现 了 一 个 博客 系统 ， 其 中 包含 所 有 基本 功能 的 项 
目 开发 ， 使 读者 对 如 何 开发 一 个 博客 网 站 有 了 一 个 初步 的 了 解 。 最 后 ， 希 望 读 者 通过 自己 的 努力 ， 来 
逐步 完善 和 加 强 这 个 博客 网 的 实用 功能 ， 最 终 成 为 一 个 令 自己 满意 的 作品 。 


a 


企业 快 信 系 统 


(Apache+PHP+Access+VWeb Service 实现 ) 
( 句 " 视频 讲解 : 1 小 时 27 分钟) 


随 着 互联 网 的 迅猛 发 展 ， 人 与 人 之 间 的 沟通 方式 也 在 发 生 着 翻天 履 地 的 变化 ， 
不 再 是 单纯 的 面对面 的 交流 、 书 信之 间 的 往来 或 者 电话 沟通 ， 而 是 选择 通过 网 络 进 
行 即 时 沟通 。 短 信和 E-mail 已 经 成 为 时 下 人 与 人 之 间 沟 通 的 另 一 座 桥 梁 。 为 此 ， 越 
来 越 多 的 网 站 都 开始 提供 发 送 E-mail 和 手机 短信 的 功能 。 短 信和 E-mail 以 其 方便 
快捷 、 零 距离 、 低 成 本 等 优势 受到 众多 企业 的 青 睦 ,成 为 企业 移动 商务 的 首先 方式 。 
本 章 的 企业 快 信 系统 就 是 讲解 如 何在 企业 中 实现 短信 和 E-mail 群发 的 功能 。 
通过 学 习 本 章 ， 读 者 可 以 学 习 到 : 


| 


豆 豆 吾 吾 吾 至 


了 解 企 业 快 信 系 统 的 开发 流程 

掌握 Access 数据 库 的 创建 方法 

掌握 PHP 如 何 通 过 ADO 方式 来 连接 Access 数据 库 
掌握 使 用 Web Service 发 送 短信 的 方法 

熟悉 POP3 服务 路 的 安装 与 配置 

熟悉 SMTP 服务 路 的 安装 与 配置 

掌握 使 用 imap 电子 邮件 系统 图 数 发 送 和 接收 邮件 的 方法 
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9.1 开发 背景 


随 着 企业 信息 化 的 飞速 发 展 ， 市 场 瞬息 万 变 ， 作 为 企业 必须 能 够 随时 掌握 市 场 的 变化 ， 针 对 其 变 
化 作出 判断 ， 并 且 能 够 第 一 时 间 将 决策 进行 传达 。 因 此 ， 急 需 一 套 既 能 从 多 方 获取 到 信息 ， 又 能 将 信 
息 进行 准确 传达 的 企业 信息 管理 系统 。 企 业 快 信 系统 就 是 信息 化 时 代 发 展 的 产物 ， 不 但 可 以 从 外 部 获 
取 到 大 量 的 数据 信息 ， 而 且 可 以 将 内 部 的 信息 传达 出 去 。 


9.2 系统 分 析 


9.2.1 需求 分 析 


企业 为 适应 市 场 的 发 展 ， 实 现 更 加 人 性 化 的 管理 ， 加 强 企 业内 部 之 间 、 企 业 与 外 部 之 间 的 沟通 ， 
为 此 需要 开发 一 套 完整 的 企业 快 信 管理 系统 。 根 据 企业 的 实际 需求 ， 企 业 快 信 系统 需要 提供 短信 和 群发、 
邮件 收发 等 功能 。 通 过 对 企业 日 常 业务 的 考察 、 分 析 ， 并 结合 短信 及 邮件 自身 的 特点 ， 得 出 本 系统 要 
求 具有 以 下 功能 : 
实现 对 客户 和 同事 信息 的 管理 。 
实现 对 常用 短语 及 其 类 别 进行 管理 。 
实现 短信 群发 的 功能 。 
实现 邮件 的 发 送 和 接收 功能 。 


办 多 办 凶 


9.2.2 可 行 性 分 析 


开发 任何 一 个 项 目 ， 首 要 任务 就 是 进行 需求 分 析 ， 然 后 根据 需求 分 析 的 结论 再 进行 项 目的 可 行 性 
分 析 ， 从 而 降低 项 目 开发 风险 ， 避 免 人 力 、 物 力 和 财力 的 浪费 ， 最 终 圆满 完成 项 目的 开发 。 项 目 开发 
的 可 行 性 分 析 主 要 从 两 个 方面 进行 : 一 方面 是 经 济 可 行 性 ， 另 一 方面 是 技术 可 行 性 。 

1， 经济 可 行 性 

采用 短信 和 邮件 作为 企业 的 移动 通信 手段 ， 将 给 企业 对 内 、 对 外 进行 信息 传递 与 沟通 带 来 革命 性 
的 变化 ， 从 而 使 得 移动 办 公 、 客 户 服务 、 员 工 沟通 等 运作 效率 显著 提升 ， 而 成 本 则 显著 下 降 。 合 理 地 
发 挥 短信 和 邮件 的 作用 ， 实 现 资源 的 充分 利用 ， 从 而 降低 企业 的 成 本 。 

2. 技术 可 行 性 

企业 快 信 系统 的 开发 主要 应 用 到 Web Service、POP3 服务 器 、SMTP 服务 器 和 imap 电子 邮件 系统 
函数 。 首 先 完 成 配置 POP3 服务 器 和 SMTP 服务 器 的 安装 和 配置 工作 ， 然 后 合理 地 运用 Web Service 和 
imap 电子 邮件 系统 函数 就 可 以 顺利 完成 该 项 目的 开发 。 
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93 系统 设计 


9.3.1 系统 目标 


根据 前 面 所 作 的 需求 分 析 和 对 企业 进行 的 实际 考察 ， 现 制定 出 企业 快 信 系统 应 达到 的 目标 。 
界面 设计 美观 大 方 。 

操作 灵活 、 方 便 。 

提供 完整 的 客户 和 同事 信息 ， 方 便 进 行 短信 和 邮件 的 发 送 。 

提供 短信 和 邮件 的 群发 功能 。 

为 短信 发 送 提供 常用 短语 。 

对 输入 的 数据 进行 严格 的 数据 检验 ， 避 免 出 现 数据 录入 的 错误 。 

系统 运行 稳定 、 数 据 库 安全 可 靠 。 


办 办 办 办 多 多 


9.3.2 ”系统 功能 结构 


根据 企业 快 信 系统 的 特点 ， 可 以 将 其 分 为 系统 信息 管理 、 邮 件 管理 、 发 送 短信 、 修 改 密码 、 帮 助 
和 退出 系统 6 个 部 分 ， 各 个 部 分 及 其 包括 的 功能 如 图 9.1 所 示 。 


EO 


邮件 管理 查找 邮件 


图 9.1 系统 功能 结构 图 
9.3.3 ”系统 流程 图 
企业 快 信 系统 的 系统 流程 如 图 9.2 所 示 。 
9.3.4 ”系统 预览 


企业 快 信 系统 由 多 个 程序 页 面 组 成 ,下面 仅 列 出 儿 个 典型 的 页 面 , 其 他 页 面 可 参见 光盘 中 的 源 程 序 。 
系统 登录 页 面 如 图 9.3 所 示 ， 该 页 面 用 于 实现 管理 员 登 录 。 主 页 面 如 图 9.4 所 示 ， 该 页 面 用 于 显示 
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系统 导航 、 操 作业 务 流程 和 版 权 信 息 等 功能 。 


[天 受 查看 邮件 | 一 "| 好 件 发 送 记 录 |->| 邮件 日 志 | 


编写 邮件 内 容 获取 邮件 地 址 发 送 邮 件 


客户 同事 信息 选择 客户 同事 信息 


EE 
ei] 


图 9.2 系统 流程 图 


是 企 业 快 信 Vi.0 


图 9.3 系统 登录 页 面 (光盘 \TMVW9\qykx\index.php) 图 9.4 主页 面 (光盘 \TM\09\qykx\indexs.php) 


发 送 短信 页 面 如 图 9.5 所 示 ， 该 页 面 用 于 实现 将 短信 息 同时 发 给 多 个 接收 者 的 功能 ， 同 时 ， 为 方便 
用 户 还 提供 了 从 客户 及 同事 列表 中 选择 接收 者 及 从 常用 短语 中 选择 指定 信息 的 功能 。 发 送 邮 件 页 面 如 
图 9.6 所 示 ， 该 页 面 用 于 实现 将 邮件 同时 发 给 多 个 接收 者 的 功能 ， 同 时 ， 为 方便 用 户 还 提供 了 从 客户 及 
同事 列表 中 选择 接收 者 的 功能 。 

接收 邮件 的 页 面 如 图 9.7 所 示 , 该 页 面 用 于 实现 邮件 信息 的 接收 功能 , 并 且 可 以 实现 删除 和 查看 指 
定 邮件 的 操作 。 查 看 邮件 内 容 的 页 面 如 图 9.8 所 示 ， 该 页 面 用 于 实现 对 指定 邮件 信息 的 输出 ， 包括 邮件 


o 
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的 主题 、 内 容 、 附 件 和 发 送 人 等 信息 。 


让 业 快 信 V1.0 本 守业 快 售 v1.0 


Bm [iD 
的 
entret smoeess? 兴 


数 际 人 Lae 
闻 wpm dsm ee int 本 旧 


[ED 4 


图 9.7 接收 邮件 的 页 面 (光盘 \TMWO9Wqylkx\lookmailphp) ”图 9.8 发 送 邮件 页 面 (光盘 \TM\09\qykx\sendmail.php) 
9.3.5 ”开发 环境 


在 开发 企业 快 信 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
.服务器 端 

操作 系统 : Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.0。 

SMTP 服务 器 。 

POP3 服务 器 。 

PHP 软件 : PHP 5.0。 

数据 库 : Access 2003。 

开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效果 1024X768 像素 。 


二 


办 办 办 办 办 办 办 多 
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客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 


9.3.6 ”文件 夹 组 织 结构 


加 回 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架 构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 
的 分 类 ， 分 别 放置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 可 以 确保 网 站 文件 目录 明确 、 
条 理 清晰 ， 同 样 也 便于 网 站 后 期 的 更 新 和 维护 。 企 业 快 信 的 文件 夹 组 织 架构 如 图 9.9 所 示 。 


日 由 vr 一 一 一 一 一 一 一 一 一 一 一 一 文件 根 目录 文件 夹 
一 一 一 一 一 一 一 一 一 一 一 并 过 娄 所 库 文件 存 人 位 置 
一 一 一 一 一 一 一 一 一 一 一 0s 样式 文件 存储 位 置 
一 一 一 一 一 一 一 一 一 一 heeess 数 据 库 文 件 存储 位 置 
一 一 一 一 一 一 一 一 一 一 一 图 片 文件 存储 的 位 置 
一 一 一 一 一 一 一 一 一 邮件 收发 系统 文件 夹 
一 一 一 一 一 一 一 一 一 一 一 发送 短信 自用 的 类 文件 
st 一 一 一 一 一 一 一 一 一 一 一 一 通过 socket 编 程 发 送 短 信 文 件 夹 


图 -图 -图 -图 -图 -图 -图 


图 9.9 企业 快 信 文 件 夹 组 织 结构 
9.4 SMTP 和 POP3 服务 器 的 安装 与 配置 


想 要 实现 邮件 的 收发 功能 就 必须 对 SMTP 服务 器 和 POP3 服务 器 进行 安装 和 配置 ， 下 面 就 来 讲解 
一 下 这 两 个 服务 器 的 安装 和 配置 过 程 。 


9.4.1 SMTP 服务 器 的 安装 和 配置 


SMTP 是 简单 邮件 传输 协议 ， 提 供 客户 端 向 服务 器 端 发 送 邮件 的 功能 ， 即 客户 端 向 服务 器 端 发 出 
请 求 指令 , 服务 器 端 则 给 出 应 答 。 本 实例 将 以 Windows 2003 Server 为 例 介 绍 SMTP 服务 器 的 安装 方法 。 

(1) 打开 “控制 面板 ”， 选 择 “添加 或 删除 程序 ”选项 ， 打 开 如 图 9.10 所 示 的 “添加 或 删除 程 
序 ” 窗 口 。 

(2) 单 击 图 9.10 所 示 窗 口中 的 “添加 /删除 Windows 组 件 ” 按 钮 ， 打 开 如 图 9.11 所 示 的 “Windows 
组 件 向 导 ” 窗 口 。 该 窗口 中 列 出 系统 的 所 有 组 件 ， 如 果 “ 应 用 程序 服务 器 ”选项 组 前 面 的 复 选 框 已 经 
处 于 选中 状态 ， 说 明 该 组 件 已 经 被 安装 ， 否 则 需要 选中 安装 。 

(3) 选中 “应 用 程序 服务 器 ” 复 选 框 后 ， 单 击 “ 详 细 信息 ”按钮 将 打开 如 图 9.12 所 示 的 “应 用 程 
序 服务 器 ”窗口 。 

(4) 选中 图 9.12 所 示 窗 口中 的 “Intemet 信息 服务 (IS) ” 复 选 框 ， 单 击 “ 确 定 ”按钮 后 开始 
安装 。 
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Nicrosoft Office 2000 Freim 
运程 管理 I 具 


9.10 “添加 或 删除 程序 ”窗口 
[windows 组 牢 身 导 E 
Windews 组 件 


图 9.11 Windows 组 件 向 导 图 9.12 应 用 程序 服务 器 


(5) 安装 完毕 后 ， 还 需要 对 SMTP 服务 器 进行 设置 ， 选 择 控 制 面板 中 的 “Intemet 信息 服务 (IS) ” 
选项 将 打开 如 图 9.13 所 示 的 “Internet 信息 服务 (IIS〉 管理 ”窗口 ， 需 要 在 该 窗口 中 创建 新 域 ， 因 为 
SMTP 只 能 向 SMTP 服务 器 中 已 经 存在 的 域名 范围 发 送 邮件 。 

(6) 右 击 图 9.13 中 的 “ 域 ”选项 ， 在 弹出 的 快捷 菜单 中 选择 “新 建 ”/“ 域 ”命令 ， 如 图 9.14 所 示 。 

me le 


西口 百 


图 9.13 SMTP 虚拟 服务 器 图 9.14 新 建 域 
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(7) 打开 如 图 9.15 所 示 的 “新 建 SMTP 域 向 导 ” 窗 口 ， 选 中 “别名 ” 单 选 按钮 ， 单 击 “ 下 一 步 ” 
按钮 ， 打 开 如 图 9.16 所 示 的 窗口 ， 在 该 窗口 中 输入 要 发 送 邮 件 的 域名 ， 单 击 “ 完 成 ”按钮 ， 即 可 实现 
SMTP 服务 器 的 配置 。 


新 建 SMTP 域 向 导 x 
欢 四 合用 V 汗 SMP 项 和 | 
i SNIP 虚拟 服务 器 上 他 娃 一 此 域 接收 的 邮件 的 地 址 宝 间 (W| 加 microsoft. con) 
指定 域 类 型 : 
个 运 妇 旭 名 称 加 : 


“BE [Ex 


Esvr-sm)] we | <tsw[ RR] mw | 
图 9.15 ”新建 SMTP 域 向 导 图 9.16 添加 域名 


9.4.2 POP3 服务 器 的 安装 和 配置 


POP 全称 为 Post Office Protocol， 邮 局 协议 ) 用 于 电子 邮件 的 接收 ， 现 在 常用 第 3 版 ， 因 此 称 为 


POP3。 通过 POP 协议 ， 客 户 机 登录 到 服务 器 后 ， 可 以 对 自己 的 邮件 进行 删除 ， 或 是 下 载 到 本 地 。POP3 
的 常用 命令 如 表 9.1 所 示 。 
表 9.1 POP3 常用 命令 
命令 参 。 数 状态 描述 
USER 此 命令 与 下 面 的 pass 命令 若 都 发 送 成 功 ， 将 使 状态 转换 
PASS 向 POP3 服务 器 发 送 指定 邮箱 的 登录 密码 ， 进 行 密码 验证 
APOP Name . Digest 认可 Digest 是 MD5 消息 摘要 
STAT 请 求 服务 器 发 回 关于 邮箱 统计 资料 (邮件 总 数 和 总 字 节 数 ) 
UIDL Msg# 处 理 回 送 邮件 唯一 标识 符 
LIST [Mse#] 处 理 梧 送 邮件 数量 和 每 个 邮件 的 大 小 
RETR [Mse#] 处 理 梧 送 由 参数 标识 的 邮件 的 全 部 文本 
DELE Msg# 处 理 服务 器 将 由 参数 标识 的 邮件 标记 为 删除 ， 由 QUIT 命令 执行 
RSET None 处 理 服务 器 将 重 置 所 有 标记 为 删除 的 邮件 ， 用 于 撤销 DELE 命令 
TOP Msg# 处 理 服务 器 将 回 送 由 参数 标识 的 邮件 前 n 行内 容 ，n 是 正 整数 
NOOP None 处 理 服务 器 返回 一 个 肯定 的 响应 ， 不 做 任何 操作 
QUIT None 更 新 断 开 与 POP3 的 连接 ， 同 时 将 带 有 删除 标记 的 邮件 进行 物理 删除 


了 解 POP3 协议 及 其 常用 命令 后 ， 下 面 就 来 介绍 POP3 服务 器 的 安装 与 配置 。 
(1) 选中 “Windows 组 件 向 导 ” 窗 口中 的 “电子 邮件 服务 ” 复 选 框 ， 单 击 “ 下 一 步 ” 按 钮 即 可 完 
成 POP3 服务 器 的 安装 ， 如 图 9.17 所 示 。 


全 注意 : 打开 “Windows 组 件 向 导 ” 窗 口 的 方法 请 详 见 实例 “SMTP 服务 器 的 安装 与 配置 ”。 
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图 9.17 “Windows 组 件 向 导 ” 窗 口 
(2) 完成 POP3 服务 器 的 安装 后 还 需要 对 服务 器 进行 设置 。 打 开 POP3 服务 器 后 ， 右 击 本 机 服务 


器 名 (如 MRZTS) ， 在 弹出 的 快捷 菜单 中 选择 “新 建 ”/“ 域 ”命令 ， 打 开 “ 添 加 域 ”对 话 框 ， 如 
图 9.18 所 示 。 在 “添加 域 ” 对 话 框 中 添加 一 个 新 域名 后 单 击 “确定 ”按钮 ， 新 域 将 被 创建 成 功 。 


图 9.18 “添加 域 ”对 话 框 


(3) 新 域 创建 完成 后 ， 需 要 在 其 中 创建 邮箱 ， 选 择 新 创建 的 域名 后 单 击 鼠 标 右键 ， 从 弹出 的 快捷 
菜单 中 依次 选择 “新 建 ”/“ 邮 箱 ” 命 令 后 ， 将 弹出 如 图 9.19 所 示 的 “添加 邮箱 ”对 话 框 ， 在 此 输入 邮 
箱 名 称 和 密码 后 即 实现 邮箱 的 创建 工作 ， 并 可 以 在 图 9.20 所 示 的 窗口 中 查看 创建 结果 。 


9.19 添加 邮箱 
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厂 不 再 显示 此 消息 


图 9.20 查看 邮箱 创建 结果 
9.5 数据 库 设 计 
Microsoft Access 是 一 种 关系 型 数据 库 管 理 系统 , 用 于 中 小 企业 管理 , 不 仅 可 以 应 用 于 本 地 数据 库 ， 
而 且 可 以 应 用 于 网 络 环境 。 
9.5.1 数据 库 概念 设计 


根据 以 上 各 节 对 系统 所 进行 的 需求 分 析 、 系 统 设 计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实 体 对 象 分 别 
为 客户 信息 实体 、 同 事 信息 实体 、 管 理 员 信息 实体 、 短 信 信 息 实体 、 邮 件 信息 实体 和 常用 短语 信息 实体 。 

1， 客户 信息 实体 

客户 信息 实体 包括 姓名 、 地 址 、 电 话 、 生 日 、 邮 箱 和 类 别 。 客 户 信息 实体 的 E-R 图 如 图 9.21 所 示 。 


图 9.21 客户 信息 实体 E-R 图 


2. 同事 信息 实体 

同事 信息 实体 包括 姓名 、 地 址 、 电 话 、 生 日 、 邮 箱 和 类 别 。 同 事 信息 实体 的 E-R 图 如 图 9.22 所 示 。 
3. 管理 员 信息 实体 

管理 员 信息 实体 包括 管理 员 名 和 管理 员 密 码 两 个 属性 。 管 理 员 信息 实体 的 E-R 图 如 图 9.23 所 示 。 
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图 9.22 同事 信息 实体 E-R 


Ce 


图 9.23 管理 员 信息 实体 E-R 图 


4. 短信 信息 实体 
短信 信息 实体 包括 他 地 址 、 发 信 手 机 号 码 、 收 信 手 机 号 码 、 短 信 内 容 、 发 信 时 间 和 短信 标题 等 属 
性 。 短 信 信 息 实 体 的 E-R 图 如 图 9.24 所 示 。 


发 信 时 间 


图 9.24 短信 信息 实体 E-R 
5. 邮件 信息 实体 


邮件 信息 实体 包括 他 地址、 发 件 邮箱 、 收 件 邮箱 、 邮 件 标 题 和 发 件 时 间 等 属性 。 邮 件 信 息 实 体 的 
E-R 图 如 图 9.25 所 示 。 


9.25 ”邮件 信息 实体 E-R 
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6. 常用 短语 信息 实体 
常用 短语 信息 实体 包括 短语 内 容 和 短语 类 别 两 个 属性 。 常 用 短语 信息 实体 的 E-R 图 如 图 9.26 所 示 。 


常用 短语 信息 短语 类 别 


图 9.26 短信 信息 实体 E-R 


9.5.2 ”创建 数据 库 及 表 


本 项 目 使 用 Access 2003 数据 库 , 数据 库 名 称 为 db_mail shortnote。 下 面 详 细 介绍 数据 库 的 创建 过 程 。 
1. 创建 数据 库 


首先 启动 Access 2003， 然 后 创建 名 为 db_mail shortnote 的 数据 库 。 创 建 数据 库 的 具体 步骤 如 下 : 
(1) 选择 “开始 ”/“ 所 有 程序 ”/Microsoft Office/Microsoft Office Access 2003 命令 ， 打 开 Access 
2003 数据 库 的 运行 环境 ， 在 此 环境 中 单 击 “ 新 建 ” 按 钮 进行 数据 库 的 创建 ， 如 图 9.27 所 示 。 


开始 工作 


Microsoft Office 
Online 


9.27 打开 Microsoft Access 


(2) 首先 在 右 侧 将 弹出 “新 建文 件 ” 列 表 项 ， 在 此 选择 “ 空 数据 库 ” 命 令 ， 然 后 手动 选择 新 建 数 
据 库 的 存储 位 置 ， 如 图 9.28 所 示 。 


视图 WD 插入 G) 工具 CD 窗口 WD 帮助 只 
PT TE TESL HEN Ee 


9.28 创建 数据 库 
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(3) 数据 库 的 存储 位 置 选择 完成 后 ， 首 先 需要 在 “文件 名 ”文本 框 中 输入 新 建 数据 库 的 名 称 ， 如 
db_mail shortnote， 该 名 称 为 本 实例 的 数据 库 名 称 ， 然 后 通过 单 击 “ 创 建 ”按钮 完成 数据 库 的 创建 ， 如 
图 9.29 所 示 。 


图 9.29 成 功 创建 数据 库 

2. 创建 数据 表 

根据 在 数据 库 概念 设计 中 给 出 的 数据 库 实体 E-R 图 ， 可 以 设计 数据 表 结 构 。 使 用 Access 数据 库 创 
建 数据 表 的 具体 步骤 如 下 : 

(1) 双击 打开 新 创建 的 db_mail shortnote 数据 库 。 

(2) 在 该 数据 库 中 ,双击 “使 用 设计 器 创建 表 ” 选 项 后 ， 进 行 新 数据 表 的 创建 。 此 时 将 打开 创建 
新 数据 表 的 对 话 框 ， 如 图 9.30 所 示 ， 在 此 输入 所 需要 创建 的 字段 名 称 ， 并 通过 “数据 类 型 ”下 拉 列 表 
来 选择 字段 所 属 的 类 型 。 

(3) 输入 完 所 需 创建 的 字段 后 ， 选 择 “文件 ”/“ 保 存 ” 命 令 ， 在 弹出 的 “另存 为 ”对 话 框 中 输 
入 所 需要 的 数据 表 名 称 即 可 ， 如 tb_user， 如 图 9.31 所 示 。 


口 表 


图 9.30 创建 新 数据 表 图 9.31 保存 新 数据 表 
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(4) 单 击 “ 确 定 ”按钮 完成 数据 表 的 创建。 
9.5.3 数据库 罗 辑 结构 设计 


企业 快 信 系 统 中 应 用 的 db_mail_ shortnote 数据 库 主 要 包含 6 个 数据 表 ， 为 了 使 读者 对 本 系统 数据 
库 中 的 数据 表 有 一 个 更 清晰 的 认识 ， 笔 者 设计 一 个 数据 表 树 形 结构 图 ， 如 图 9.32 所 示 。 


db_mail_shortnote 


加 使 用 设计 器 创 寻 表 

ell 使 用 向 导 创 娃 表 

合 通过 输入 数据 创 哇 表 

日 tbcustmer 一 一 一 一 客户 信息 表 
加 tb_colleague 一 一 一 一 一 一 一 一 一 同事 信息 表 

日 tn 一 一 一 一 一 一 一 一 一 一 邮件 信息 表 
日 th_short 一 一 一 一 一 一 一 一 一 一 短信 信息 表 
加 th_jmote 一 一 一 一 一 一 一 一 一 一 一 常用 短语 信息 表 
可 thnser 一 管理 员 信 息 表 


图 9.32 数据 表 树 形 结构 图 

各 数据 表 的 逻辑 结构 如 表 9.2 一 表 9.7 所 示 。 

1. tb_user (管理 员 信息 表 ) 

管理 员 信息 表 主要 用 于 保存 管理 员 的 基本 信息 ，tb_user 表 的 结构 如 表 9.2 所 示 。 
表 9.2 tb_user 表 的 结构 


唯一 标识 


2. tb_colleague (同事 信息 表 ) 
同事 信息 表 主 要 用 于 保存 企业 中 员工 的 个 人 信息 ，tb_colleague 表 的 结构 如 表 9.3 所 示 。 
表 9.3 tb_colleague 表 的 结构 


字段 名 称 
colleague id | 唯一 标识 
colleague_ name | 姓名 
colleague_address | 地 址 
colleague tel 电话 


colleague mail 
colleague_birthdar 
colleague_categol 


3. tb_customer (客户 信息 表 ) 
客户 信息 表 主 要 用 于 保存 与 企业 长 期 合作 关系 的 客户 的 信息 ,tb_customer 表 的 结构 如 表 9.4 所 示 。 
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表 9.4 tb_customer 表 的 结构 


字段 名 称 


customer id 


姓名 


customer name 


customer _ address 


customer tel 


customer_mail 


customer_birthday 


并 | 到 | 码 | 碟 | 咏 | 半 


customer categol 


4. tb_mail (邮件 信息 表 ) 
邮件 信息 表 主 要 用 于 保存 邮件 的 发 送 记 录 ，tb_mail 表 的 结构 如 表 9.5 所 示 。 
表 9.5 tb_mail 表 的 结构 


段 名 称 字段 描述 
mail id 唯一 标识 
mail i IP 地 址 
mail formuser 发 件 人 
mail touser 收 件 人 
mail title 标题 
mail date 发 件 时 间 


5. tb_short (短信 信息 表 ) 
短信 信息 表 主 要 用 于 存储 短信 的 发 送 记 录 ，tb_short 表 的 结构 如 表 9.6 所 示 。 
表 9.6 tb_short 表 的 结构 


short id 唯一 标识 
short ip | | 是 | ep 地 址 
short_tel | ”文本 | 。 是 | ”发 送 手 机 号 码 
short tels | ”文本 | 和 否 ”| 接收 手机 号 码 
short content | 文本 | 是 | 短信 内 容 
short date | 日 期 时 间 | 否 | 发送 时 间 


文本 
6. tb_note (常用 短语 信息 表 ) 


short title 标题 


常用 短语 信息 表 主 要 用 于 存储 常用 短语 的 内 容 ，tb_note 表 的 结构 如 表 9.7 所 示 。 
表 9.7 tb_note 表 的 结构 


字段 名 称 


note id 


note_content 
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续 表 
字段 名 称 数据 类 型 多 许 空 字段 描述 
note catego 文本 50 是 短语 类 别 


9.6 首页 设计 


9.6.1 首页 概述 


管理 员 要 通过 用 户 名 和 密码 登录 到 企业 快 信 系统 的 主页 面 。 系 统 主页 面 主要 包括 系统 导航 栏 、 功 
能 展示 区 和 信息 选择 区 3 部 分 。 其 中 ， 导 航 栏 主 要 包括 连接 短信 、 连 接 邮 件 、 修 改 密码 、 帮 助 和 退出 
系统 ;功能 展示 区 主要 用 于 实现 发 送 短信 和 邮件 收发 ， 以 及 对 系统 信息 管理 的 操作 ;在 信息 选择 区 中 
以 树 状 菜单 的 形式 展示 出 系统 中 的 客户 信息 、 同 事 信 息 和 系统 信息 管理 的 内 容 。 

下 面 看 一 下 本 案例 中 提供 的 首页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\09\qykx\index.php， 如 
图 9.33 所 示 。 


可 欢迎 : 欢迎 您 使 用 企业 快 信 癌 理 系统 。 
添加 手机 号 : 添加 | 


发 送 的 手机 号 码 
量 除 


短 东 内 容 


注册 手机 号 。 号 


im [ ee 
ET 信息 选择 区 


I 发 信号 码 收 六 号码 短信 内容 发 送 时 间 
192, 168,1 59 13604338784 sdfsd sd 2007-12-8 501:16 扣除 
192.168.1.59 13604336784 afeifedfg 。 2007-12-6 6:04:41 蝇 除 
发 送信 息 2 条 每 页 显示 5 条 第 1 页 / 共 ! 页 ' 


图 9.33 企业 快 信 首 页 
9.6.2 首页 技术 分 析 


在 图 9.33 所 示 的 首页 中 ， 系 统 导航 栏 和 版 权 信 息 栏 是 固定 不 变 的 ， 只 是 功能 展示 区 和 信息 选择 区 
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根据 系统 提供 的 变量 不 同 而 进行 不 同 功能 模块 之 间 的 跳 转 ， 主 要 应 用 switch 语句 来 完成 。 当 单 击 导航 
栏 中 的 按钮 时 , 系统 就 会 提交 一 个 变量 值 给 switch 语句 , 然后 switch 语句 会 根据 提供 的 变量 进行 判断 ， 
最 终 输 出 指定 的 模块 。 

应 用 switch 语句 实现 不 同 功能 模块 之 间 的 跳 转 ， 关 键 代 码 如 下 : 


例 程 01 ”代码 位 置 : 光盘 \TMWO9qylesindexs php 


<?php 
switch($lmbs){ /| 执行 switch 语 句 ， 根 据 提交 的 变量 值 
case "连接 短信 " : // 判 断 提 交 的 变量 是 否 与 连接 短信 匹配 
include "short_note.php"; 。 // 如 果 相 匹配 ， 则 执行 include 包 含 语句 ， 调 用 short_note.php 文 件 
break: /跳出 循环 
case "连接 邮件 ": 
include "mail php": 
break:; 
/这 里 省 略 了 部 分 代码 
case ™: // 当 变量 值 为 空 时 
include "short_note.php"; /调用 该 文件 
break: 
} 
?> 


9.6.3 首页 的 实现 过 程 


国 ”首页 使 用 的 数据 表 : tb_customer、tb_colleague、tb_mail 、tb_note、tb_short 

主页 面 的 实现 过 程 可 以 归结 为 3 个 部 分 ， 首 先 判 断 登 录用 户 是 否 为 管理 员 ， 如 果 不 是 管理 员 则 不 
能 进入 到 该 系统 ， 然 后 设置 主页 面 的 导航 栏 ， 最 后 通过 switch 语句 实现 不 同 功能 模块 之 间 的 跳 转 。 

判断 登录 用 户 权限 的 原理 是 获取 登录 用 户 的 用 户 名 ， 如 果 用 户 名 变量 存在 ， 则 说 明 该 用 户 是 管理 
员 ， 和 否则 该 用 户 就 不 是 正确 登录 ， 将 弹出 提示 信息 “您 没有 正确 登录 ”， 返 回 到 系统 的 登录 页 面 。 关 
键 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 \TMVW9\Wqykx\indexs php 


<?php 
session start(): // 初 始 化 session 变 量 
include_once("conn/conn.php"): // 连 接 数 据 库 
这 $username 一 true){ /判断 用 户 登 录 的 变量 是 否 为 真 
> 
<!-> 这 里 省 略 了 执行 内 容 <!-> 
<?php 
}else{ // 如 果 变量 不 为 真 则 执行 下 面 的 代码 


echo "<script>alert(' 您 没有 正确 登录 !"):window.location href='index.php':</script>"; 


时 
?> 


主页 面 的 导航 栏 是 通过 设置 超 链接 来 完成 的 ， 并 且 为 超 链 接 设 置 变 量 标识 符 ， 指 定 不 同 的 变量 值 ， 
用 于 实现 不 同 功能 模块 之 间 的 跳 转 。 关 键 代 码 如 下 : 
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例 程 03 ”代码 位 置 : 光盘 \TM\09\qykx\indexs.php 
<!--> 指 定 连接 到 indexs.php 文 件 ， 使 用 变量 标识 符 为 Imbs<!--> 
<td width="78" align="center"> 
<a href-"indexs.php?lmbs= 连 接 短信 " class="STYLE1"> 连 接 短信 </a> 
</td> 
<td width="75" align="center"> 
<a hre 人 "indexs php?lmbs= 连 接 邮 件 " class="STYLE1"> 连 接 邮件 </a></td> 
<!--> 这 里 使 用 的 是 onClick 事 件 ， 调 用 一 个 弹出 窗口 ， 并 且 指 定 窗口 的 大 小 <!--> 
<td width="77" align="center"> 
<a href="#" class="STYLE1" 
onClick="MM _openBrWindow(update pwd.php',".'toolbar=yes,width=440,height=219")"> 修 改 密码 </a> 
</td> 


最 后 通过 switch 语句 实现 不 同 模块 之 间 的 跳 转 ， 其 关键 代码 已 经 在 主页 面 技术 分 析 中 给 出 ， 这 里 
不 再 獒 述 。 


9.7 系统 信息 管理 模块 设计 


9.7.1 系统 信息 管理 模块 概述 


系统 信息 管理 模块 主要 实现 对 客户 信息 、 同 事 信息 和 常用 短语 的 管理 。 其 中 客户 信息 管理 包括 添 
加 客户 信息 、 删 除 客户 信息 、 客 户 信息 的 显示 和 提交 客户 信息 ; 同事 信息 管理 包括 添加 同事 信息 、 删 
除 同 事 信息 、 同 事 信息 的 显示 和 提交 同事 信息 ; 常用 短语 管理 包括 添加 常用 短语 信息 、 删 除 常 用 短语 、 
常用 短语 的 显示 和 提交 常用 短语 。 系 统 信息 管理 模块 的 功能 结构 如 图 9.34 所 示 。 


系统 信息 管理 


了 了 遇 车 强 
3 
如 可 车 强 
姗 可 顶 哩 
济 洁 对 疡 
3 


图 9.34 系统 信息 管理 模块 的 功能 结构 
9.7.2 ”系统 信息 管理 模块 技术 分 析 
在 系统 信息 管理 模块 中 有 这 样 一 个 功能 ， 将 系统 中 客户 、 同 事 的 电话 或 者 邮箱 地 址 直接 提交 到 指 
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定 的 位 置 ， 便 于 手机 号 码 和 邮箱 地 址 的 选择 。 该 功能 的 实现 主要 应 用 session 变量 ， 其 实现 原理 与 购物 
车 是 相同 的 。 首 先 从 数据 库 中 读 取 到 指定 的 电话 号 码 ， 然 后 将 选择 的 电话 号 码 添加 到 一 个 session 变量 
中 进行 存储 ， 最 后 在 短信 发 送 功能 模块 中 读 取 session 变量 中 存储 的 电话 号 码 。 

这 里 以 短信 发 送 模块 中 的 客户 电话 号 码 的 选择 为 例 进 行 讲解 ， 首 先 从 数据 库 中 读 取 客户 信息 ， 输 
出 客户 姓名 , 并 且 设 置 超 链接 指定 变量 标识 ides 的 值 为 客户 信息 的 ID, 将 变量 值 提交 到 che4.php 页 中 。 
关键 代码 如 下 : 


例 程 04 ”代码 位 置 : 光盘 VTMWO9wylesselectphp 


<?php 
$sql="select * from tb_customer"; // 读 取 数 据 库 中 数据 
Srs=new com("adodb.recordset"): /应 用 类 
Srs->open($sql.$conn.1.1): // 执 行 查询 语句 
while(!S$rs->eof){ // 循 环 输出 读 取 到 的 数据 
Sfields=Srs->fields(customer id): /变量 赋值 
> 


<tr> 
<!--> 设 置 超 链 接 ， 将 常用 短语 的 ID 值 提交 到 che.php 页 中 <!--> 
<td><a href="che4.php?idss=<?php $fields=$rs->fields(customer id):echo $fields->value;?>"> 
<?php $fields=$rs->fields(customer name)j:echo $fields->value:?></a></td> 
<<!-->------------- 一 --- 一 ---- 一 -- <1--> 


</tr> 
<?php ”Srs->movenext; } ”// 语 句 执行 结束 ?> 


然后 在 che4.php 页 中 根据 获取 的 idss 变量 的 值 ， 将 客户 信息 的 ID 值 存储 到 创建 的 session 变量 
producelistss 中 。 在 将 客户 的 ID 传递 到 短信 发 送 页 之 前 ， 首 先 要 判断 ID 在 session 变量 中 是 否 已 经 存 
在 , 如 果 存 在 , 则 提示 “该 电话 号 码 已 经 被 选中 !”; 如 果 不 存在 , 则 将 值 存储 到 session 变量 producelistss 
中 保存 ， 并 且 跳 转 到 短信 发 送 页 面 中 。 程 序 的 关键 代码 如 下 : 

例 程 05 ”代码 位 置 :光盘 \TM\09\qykx\che4.php 


<?php session start(); // 初 始 化 session 变 量 
if($_SESSION[usemame]—""){ // 判 断 是 否 是 正确 登录 
echo "<script>alert( 请 先 登录 后 才 可 以 发 送 短 信 !"):history.back0:</script>"; 
exit:} 
© Sidss=strval($ GET [idss]): // 获 取 超 链接 提交 的 变量 值 
© $arrayss=explode("(@".$_SESSION[producelistss]): // 以 @ 来 分 割 session 变 量 中 的 值 
for($i=0:;$i<count($arrayss)-1:$i++){ // 循 环 读 取 数 组 中 的 值 
这 $arrayss[$i 一 Sidss){ /独断 session 变 量 中 是 否 存在 超 链接 提交 的 变量 值 
echo "<script>alert(' 该 电话 号 码 已 经 被 选中 !):history.backO:</script>"; 
exit} } 
@ $_SESSION[producelistss]=$_SESSION[producelistss].Sidss."@";  ” // 将 $idss 添 加 到 session 变 量 中 
$_SESSION[quatityss]j=$_SESSION[quatityss]."1@": 
@ header("location:indexs .php?lmbs= 连 接 短信 "): // 跳 转 到 发 送 短信 页 面 
?> 
< 代码 贴 十 


@ $ GET[idss]: 获取 超 链接 中 变量 提交 的 值 。 
@ explode0: 将 字符 囊 依 指定 的 字符 囊 或 字符 separator 切 开 。 
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array explode(string separator. string string, [int limit]) 
该 函数 返回 由 字符 串 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子囊 ， 它 们 被 字符 囊 separator 作为 边界 点 分 隔 出 来 。 
如 果 设 置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 将 包含 string 的 剩余 部 分 ; 如果 separator 
为 空 字符 串 ("") ，explode0 函 数 将 返回 false; 如 果 separator 所 包含 的 值 在 string 中 找 不 到 ， 那 么 explode0 函 数 将 返回 
包含 string 单个 元 素 的 数组 ; 如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 
目 $ SESSION[producelistss]: 定义 的 session 变量 ， 将 $idss 的 值 添加 到 session 变量 中 。 
@ header0: 实现 页 面 之 间 的 跳 转 ， 这 里 跳 转 到 发 送 短信 页 面 。 


最 后 在 发 送 短信 页 面 中 从 session 变量 中 获取 到 提交 的 电话 号 码 ， 其 中 主要 应 用 的 就 是 explode0 
函数 和 implode0 函 数 。 从 session 变量 中 获取 到 值 后 ， 以 该 值 为 条 件 执行 查询 语句 ， 从 tb_customer 表 
中 读 取 到 对 应 的 电话 号 码 ， 最 后 以 隐藏 域 的 形式 将 电话 号 码 进行 提交 。 程 序 的 关键 代码 如 下 : 

例 程 06 ”代码 位 置 : 光盘 \TMVW9\qykx\short_ note php 


<?php 
Sarrayss=explode("(@",$_SESSION[producelistss]): /以 @ 分 隔 session 变 量 producelistss 中 存储 的 数据 
Sarrayquatityss=explode("(@",$_SESSION[quatityss]); /以 @ 分 隔 session 变 量 quatityss 中 存储 的 数据 


while(list($name,$value)=each($_ POST)){ // 通 过 while 语 句 和 list 函 数 循环 读 取 出 数组 中 的 数据 
for($i=0:$i<count($arrayss)-1:$i++){ 
if(($arrayss[$il)=—=$name){ 
Sarrayquatityss[$i]=$value:; // 输 出 变量 值 


} 
} 
; 
$_SESSION[quatityss]=implode("(@",$arrayquatityss); 。 // 将 数组 $arrayquatityss 中 的 数据 以 @ 进 行 输出 
for($i=0:$i<count($arrayss)-1:$i+H+){ 
Sidss=$arrayss[$i]: // 输 出 数组 中 的 数据 
if($idss!=""){ /判断 如 果 $idss 的 值 不 为 空 
/如 果 idss 的 值 不 为 室 ， 则 执行 下 面 的 查询 语句 ， 条 件 为 customer id 的 值 等 于 Sidss 的 值 
$sql='select * 位 om tb_customer where customer id= .Sidss."… 
Srsss=new com("adodb.recordset"): 
Srsss->open($sql.$conn,3,1): // 执 行 查询 语句 
?> 
<!--> 使 用 隐藏 域 提交 客户 电话 号 码 <!--> 
<input type="hidden" name="colleague tel[]" value="<?php $fields=S$rsss->fields(customer tel):echo 
$fields->value:?>"><?php $fields=S$rsss->fields(customer tel):echo $fields->value:?> 
<!-->--------- 一 ------- 一 -一 <!--> 


<php } } ?> 
从 注意 ; 该 技术 不 只 在 获取 客户 的 电话 号 码 中 使 用 ， 在 获取 同事 电话 号 码 和 常用 短语 时 都 应 用 到 该 
技术 ， 并 且 在 邮件 发 送 模块 中 也 应 用 到 该 技术 来 获取 对 应 客户 或 者 同事 的 邮箱 地 址 。 


9.7.3 客户 信息 管理 模块 的 实现 过 程 


国 客户 信息 管理 使 用 的 数据 表 : tb_customer 
在 系统 信息 管理 模块 的 设计 中 ， 各 个 子 功能 模块 的 实现 方法 是 相同 的 。 下 面 以 客户 信息 管理 为 例 ， 
讲解 一 下 如 何 实现 客户 信息 的 添加 、 删 除 和 显示 。 客 户 信息 管理 模块 的 运行 结果 如 图 9.35 所 示 。 
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合计 1 人 每 页 是 寺 10 条 信息 第 页 / 共 ! 页 


9.35 客户 信息 管理 模块 的 运行 结果 


客户 信息 管理 模块 主要 包括 信息 添加 、 删 除 、 显 示 和 提交 4 部 分 内 容 ， 其 中 有 关 提交 使 用 的 技术 
已 经 在 技术 分 析 中 做 了 详细 的 讲解 ， 不 再 袭 述 。 这 里 主要 讲解 信息 添加 、 删 除 和 显示 的 方法 。 

客户 信息 的 添加 和 删除 操作 是 在 客户 信息 显示 的 页 面 中 完成 的 ， 客 户 信息 显示 主要 通过 
customer_gl.php 文件 来 完成 ， 在 该 页 中 对 数据 库 中 存储 的 数据 进行 输出 ， 并 且 以 分 页 的 形式 进行 显示 。 

首先 连接 数据 库 ,然后 从 数据 库 中 读 取 客户 信息 ,最 后 应 用 while 循环 语句 以 分 页 的 形式 进行 显示 。 


程序 的 关键 代码 如 下 : 
例 程 07 ”代码 位 置 : 光盘 \TM\WO9\Wqykx\customer_gLphp 
<?php 
include("conn/conn.php"): // 连 接 数 据 库 
$sql="select * from tb_customer": // 创 建 查询 语句 
Sres=new com("adodb.recordset"): /应 用 类 
@ Sres->open(Ssql,Sconn,3,1); // 执 行 查询 语句 
Sres->pagesize=10: // 定 义 变量 每 页 显示 10 条 记录 


?> 


/根据 $_GET 获 取 的 值 显 示 页 面 信息 

if((trim(intval($_GET[page]))=—"")ll(intval($ ee _GET[page])<=0){ 
Spage=1: 

jelse{ 
Spage=intval($_GET[page]): // 获 取 变量 提交 的 值 


中 
if($res->eof | Sres->bof){ // 为 空 时 输出 没有 信息 


<tr><td height="20" colspan="6" bgcolor="#F8F8F8"><div align="center" class="STYLE1"> 没 有 信息 </div></td></tr> 
<?php 


B 


}else{ 


第 9 章 企业 快 信 系统 ( ApacheHPHP+Access+Web Service 实现 ) 


$res->absolutepage=$page: /定义 变量 

Smypagesize=$res->pagesize: // 将 记录 的 值 赋 给 给 变量 

while(!$res->eof && Smypagesize>0){ /执行 while 循 环 语句 

?> 
<t> 
// 省 略 了 部 分 HTML 代 码 
<td we "center" bgcolor="#F 8F8F8"><a hre 全 "delete_ colleague_customer.php?lmbs=<?php echo 

$_GET[Imbs]:?>&customer=<?php $fields=Sres->fields(customer id):echo $fields->value:?>" class="STYLE1"> 删 除 
</a></td> 


<?php 
Smypagesize--; 
Sres->movenext; 


} 

?> 
< 人 代码 贴 二 

@ Sres->open($sql,$Sconn,3,1):: 执行 SQL 语句 ， 其 中 “3” 代 表 可 读 写 方式 ， 当 前 记录 集 指针 可 自由 移动 并 可 看 到 
新 增 记录 ; “1” 代 表 默 认 锁 定 类 型 ， 记 录 集 只 读 ， 不 能 修改 记录 。 

@ intval0 函 数 : 获取 变量 的 整数 值 。 


上 述 代码 中 实现 读 取 数 据 库 中 数据 ， 并 且 为 分 页 技术 的 实现 定义 变量 ， 有 关 分 页 的 详细 代码 可 以 
参考 本 书 的 光盘 。 

客户 信息 添加 功能 在 customer glLphp 页 中 通过 超 链 接 打开 的 窗口 中 完成 ， 其 中 主要 应 用 的 是 
onClick 事件 ， 调 用 一 个 MM _openBrWindow 函数 ， 其 中 设置 连接 文件 为 insert_customer.php， 指 定 窗 
口 的 宽度 是 440， 高 度 为 219。 关 键 代码 如 下 : 


例 程 08 ”代码 位 置 ， 光 扒 \TM\09\qykx\customer_gl.php 

<script type="text/JavaScript"> 

function MM_openBrWindow(theURL,.winName,features) {  //v2.0 
window.open(theURL.winName.features): // 设 置 弹出 窗口 

b 


</script> 

<!-- 应 用 onclick 事 件 调用 函数 ， 实 现 弹 出 窗口 --> 

<td align="center" bgcolor="#E9F7FD"><a href="#" class="STYLE1" 
‘onClick="MM_openBrWindow('insert_customer.php',",'toolbar=yes.width=440.height=219")"> 添 加 客户 </a></td> 


客户 信息 的 删除 功能 是 根据 在 customer_gl.php 页 中 获取 的 客户 信息 数据 ID, 设置 超 链 接 将 客户 信 
息 数据 ID 作为 变量 值 提交 到 delete_colleague_customer.php 页 中 执行 删除 的 操作 ， 在 delete_colleague_ 
customer.php 页 中 ， 将 客户 信息 人 D 作为 执行 删除 语句 的 条 件 。 程 序 的 关键 代码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\09\qykx\delete_colleague_customer.php 


<?php include("conn/conn.php"): // 连 接 数 据 库 
if($colleague==true){ /判断 提交 的 变量 值 是 否 为 空 
S$sql='delete from tb_colleague where colleague_id=-'.$colleague": /创建 删除 语句 
Srs=new com("adodb.recordset"): // 应 用 类 
Srs->open($sql.$conn.3.1): // 执 行 删除 语句 


echo "<script>alert( 同 事 信息 删除 成 功 !""):window.location.href='indexs.php?lmbs=$_GET[Imbs]&lmlb= 同 事 信息 
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管理 ':</script>"; 


9.7.4 ”单元 测试 


在 对 系统 信息 管理 模块 进行 测试 的 过 程 中 , 当 打 开 同 事 信息 管理 模块 时 , 输出 如 图 9.36 所 示 的 页 面 。 


姓名 电话 邮箱 生日 地 址 添加 同事 
潘 ** 11-3 长 春 市 损 际 
潘 t* 11-3 长 春 市 贡院 
潘 t+ 11-3 长 春 市 是 除 
潘 wy 13604338784 11-3 长 春 市 吉 除 
潘 t 13804338784 11-3 长 春 市 Li 
潘 * 13604336764 11-3 长 春 市 最 除 
潘 t* 13804338784 11-3 长 春 市 bd 
潘 t* 13604356784 11-3 长 春 市 山区 
潘 * 13604336784 11-3 长 春 市 ud 
潘 t* 13604358764 11-3 长 春 市 贡 除 
合 it2 人 每 页 显示 10 条 信息 第 ! 页 / 共 1 页 1 


9.36 同事 信息 管理 数据 输出 结果 


从 图 9.36 所 示 的 输出 结果 中 可 以 看 到 , 在 输出 查询 结果 的 过 程 中 将 同一 条 记录 循环 输出 10 次 。 从 
而 可 以 判断 出 程序 中 存在 一 个 逻辑 错误 , 即 在 执行 查询 语句 的 过 程 没 有 将 指针 的 位 置 指 向 下 一 条 记录 ， 
而 是 将 指针 当前 位 置 的 数据 输出 10 次 。 查 看 程序 中 的 代码 ， 出 现 错误 的 地 方 如 图 9.37 所 示 。 

PHP 是 通过 预先 定义 类 com 来 使 用 ADO 方法 操纵 Access 数据 库 的 , 在 通过 $rs->open 执行 查询 语 
名 的 过 程 中 ， 一 次 查询 结束 之 后 必须 应 用 $rs->movenext 语句 将 指针 的 位 置 指向 下 一 条 记录 ， 否 则 查询 
语句 不 会 自动 执行 下 一 查询 , 也 就 是 导致 循环 输出 10 条 相同 记录 的 原因 。 其 正确 的 代码 输出 应 该 是 在 执 
行 while 循环 语句 后 应 用 $rs->movenext 语句 将 指针 的 位 置 指向 下 一 条 记录 。 正 确 代 码 输 出 如 图 9.38 所 示 。 


<2php 《?php 

jelse{ }else{ 
$res->absolutepage=$page; $res->absolutepage=$page; 
Snypagesize=$rs-)pagesize; Snypagesize=$rs_ pasesize 


while (I$re->eof 4& $nypagesize>0) { while(1grs-yeof &h Snypagesize>0) [ 
?7 

1 一 一 一 -循环 给 出 查询 出 的 语句 一 一 
《<?php 

MA /终止 本 次 短 环 
1 
2> 


图 9.37 程序 中 出 现 的 逻辑 错误 图 9.38 正确 的 代码 输出 
9.8 发 送 短 信和 模块 设计 


9.8.1 发送 短信 模块 概述 


发 送 短信 功能 的 实现 主要 应 用 的 是 Web Service, 这 里 使 用 的 是 新 浪 网 提供 的 一 种 发 送 短信 的 Web 
Service, 有 关 Web Service 技术 将 在 技术 分 析 中 进行 详细 讲解 在 该 模块 中 不 但 可 以 进行 单条 短信 发 送 ， 
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而 且 可 以 进行 短信 群发 。 其 操作 的 流程 如 图 9.39 所 示 。 


9.8.2 ”发 送 短信 模块 技术 分 析 


在 发 送 短信 模块 中 主要 应 用 的 


是 由 新 浪 提供 的 Web Service， 地 址 是 http://smsinter.sina.com.cn/ws/ 


smswebservice0101.wsdl。 在 这 个 Service 中 提供 一 个 发 送 短 消息 的 方法 sndXml0。sendXml(0 方 法 中 的 
6 个 参数 均 为 string 类 型 ， 并 且 sendXml() 方 法 的 返回 值 也 是 string 类 型 。 该 方法 的 语法 格式 如 下 : 


string sendXml (carrier , userid , password , mobilenumber , content , msgtype ) 


参数 说 明 如 表 9.8 所 示 。 
表 9.8 sendXml() 方 法 的 参数 说 明 
参数 说 了 明 

. 运营 商 名 称 ， 具 体 使 用 时 此 参数 并 没有 什么 具体 要 求 ， 所 以 这 里 面 可 以 任意 输 写 ， 输 入 的 字符 串 也 
Ee 不 会 在 对 方 手机 中 有 任何 显示 
userid 在 新 浪 网 上 注册 的 手机 号 ， 注 册 手 机 所 用 的 地 址 是 http://sms.sina.com.cn 
password 用 户 在 新 浪 网 成 功 注册 手机 后 所 反馈 的 密码 
mobilenumber | 要 发 送 短信 到 对 方 的 手机 号 码 
content 所 要 发 送 短 消息 的 内 容 
msgtype 发 送 短 消息 的 类 型 ， 由 于 本 实例 发 送 的 不 是 彩信 ， 所 以 输入 Text 表示 短信 内 容 为 文本 

只 需要 将 上 述 参 数 中 涉及 到 的 内 容 以 表单 的 形式 提交 到 该 方法 中 ， 就 可 以 完成 短信 的 发 送 。 


9.8.3 ”短信 发 送 模块 的 实现 过 程 


国 ”短信 发 送 模块 使 用 的 数据 表 : 


tb_colleague、 tb_short、 tb _ note 


当 管 理 员 进入 到 企业 快 信 管理 系统 以 后 ， 默 认 展示 的 就 是 短信 发 送 的 功能 页 面 ， 其 运行 结果 如 
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图 9.40 所 示 。 


于 加 手机 号 : 
发 送 的 手机 号 码 


短信 内 容 


注册 手机 号 
注册 密码 


可 欢迎 : 欢迎 您 使 用 企业 快 信 管理 系统 


Ja | 


13804338764 ”135T6665613 量 脸 


年 快乐 


[EL EE rrrr EE 
[rrrrr 


9.40 短信 发 送 功能 的 运行 结果 


在 发 送 短信 模块 中 主要 应 用 两 个 文件 : 一 个 是 short_note.php 文件 ， 用 于 创建 一 个 提交 短信 内 容 的 
表单 ， 包 括 发 送 的 手机 号 码 、 短 信 内 容 、 注 册 手 机 号 码 和 注册 密码 等 内 容 ; 另 一 个 是 send.php 文件 ， 
用 于 对 表单 中 提交 的 数据 进行 处 理 ， 在 该 文件 中 将 数据 提交 到 sendXml0 方 法 中 完成 短信 的 发 送 ， 并 且 


将 短信 发 送 的 记录 存储 到 数据 库 中 。 
在 short_note.php 文件 中 创建 一 个 forml 表单 ， 用 于 提交 发 送 短信 的 相关 信息 。forml 表单 中 的 重 
要 属性 如 表 9.9 所 示 。 
表 9.9 发 送 短信 中 使 用 的 重要 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含义 
和 i ehod post action="send.php" onSubmit="javascript: return 表单 
checkitsO” 
| bidden name=" " value="<?php $fields=S$rs->fields(colleague_tel):echo 同事 电话 
Sfields->value:?>" 

. value="<?php $fields=S$rsss->fields(customer tel):echo 
colleague tel[] | 。 hidden i 客户 电话 
colleague tel[] | _ hidden value="<?php echo $ids:?>"><?php echo S$ids:?> 添加 的 电话 
Tegtel text Class="inputcss" id="regtel" size="30" 注册 手机 号 码 
regpwd password Class="inputcss" id="regpwd" size="30" 注册 密码 
Submit submit class="buttoncss" id="Submit" value=" 发 送 " 提交 表单 


在 short_note.php 文件 中 还 可 以 实现 客户 、 同 事 电 话 以 及 常用 短语 的 自动 添加 功能 ， 有 关 该 功能 的 
实现 已 经 在 系统 信息 管理 模块 设计 一 节 中 的 技术 分 析 中 进行 了 详细 的 讲解 ， 这 里 不 再 獒 述 。 


% 
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完成 表单 内 容 的 填写 后 ， 就 可 以 将 表单 中 的 内 容 提 交 到 send.php 文件 中 ， 在 该 文件 中 完成 短信 的 
发 送 ， 首 先 将 短信 的 发 送 记 录 存 储 到 数据 库 中 ， 然 后 将 数据 以 数组 的 形式 提交 到 sendXml0 方 法 中 完成 
短信 的 发 送 。 关 键 代码 如 下 : 

例 程 10 ”代码 位 置 : 光盘 \TMWO9vqylkssendphp 


<?php include once("conn/conn.php"): // 连 接 数 据 库 
这 $_ POST[colleague_ tel]—""){ // 济 断 电 话 号 码 是 否 为 空 
echo "<script>alert(' 请 填写 发 送 的 手机 号 码 !"):history.back0:</script>"; 
}else{ 
Scarrier=" 吉 林 省 明日 科技 有 限 公司 ": /获取 标题 
Suserid=trim($_POST[regtel]): // 获 取 发 送 手 机 号 码 
Spassword=trim($_POST[regpwd]); // 获 取 密 码 
S$content=trim($ POST[mess]): /获取 短信 内 容 
Sdata=date("Y-m-d H:i:s"): // 获 取 时 间 
Sip=getenv(REMOTE_ADDR'); /获取 了 下 地 址 
while(list($name,$value)=each($_POST[colleague_tel])){// 读 取 要 发 送 的 电话 号 码 
ifis_numeric(Svalue) 一 true){ // 判 断 电话 格式 是 否 正确 
S$mobilenumber=$value; // 将 获取 的 电话 号 码 附 给 变量 
$msgtype="Text"; // 指 定 短信 为 文本 格式 


刻 向 数据 库 中 添加 发 送 短信 的 记录 */ 

$sql="insert into 
tb short(short ip,short tel,short tels,short content,short date,short title)values('$ip','$userid','$mobilenumber','$content','$data', 
‘Scarrier)": 

Srs=new com("adodb.recordset"); 


Srs->open($sql,$conn,3,3): // 执 行 添加 语句 
2 a 
oo include('nusoap/lib/nusoap.php'); // 读 取 PHP 类 文件 ， 实 现 短信 的 发 送 
必 将 数据 以 数组 的 形式 添加 到 sendXml 方 法 中 所 
四 $s=new soapclient('http://smsinter.sina.com.cn/ws/smswebservice0101.wsdl'WSDL"):; 
© $s->call('sendXml',array('parameters' =>array('carrier’ => $carrier,'userid'=> $userid,'password' => 
S$password,'mobilenumber’ => $mobilenumber,'content' => $content,'msgtype' => $msgtype))): 
六 */ 
echo "<script>alert(' 短 信 发 送 成 功 !");window.location.href="indexs.php?limbs= 连 接 短 信 ';</script>"; 
让 
?> 
< 代码 贴 十 


@ include: 通过 include 语句 包含 nusoap.php 类 文件 。 
@ soapclient: 应 用 soapclient 创建 对 象 。 
四 $s->call: 应 用 call0 方 法 传递 参数 。 


上 述 代 码 中 包含 一 个 外 部 文件 nusoap.php, 该 文件 中 包含 一 组 开源 的 , 用 来 通过 HTTP 收发 SOAP 
消息 的 PHP 类 。 

通过 调用 WebService 的 方式 实现 短信 的 发 送 ， 首 先 应 该 从 http://dietrich.ganx4.con/nusoap/ 处 下 载 
nusoap php 文件， 并 在 程序 中 用 include0 函 数 将 nusoap.php 文件 包含 进来 ， 然 后 用 soapclient 类 创建 一 
个 soapclient 对 象 ， 该 类 的 初始 化 参数 设置 为 Web Service 的 URL 地 址 ， 并 通过 该 类 的 call0 方 法 调用 
Web Service 中 的 远程 方法 ，call0 方 法 中 的 参数 sendXml 是 Web Service 提供 的 远程 方法 名 ， 
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“array(parameters' =>array('carrier' => $carrier,'userid'=> $userid,'password' => S$password,'mobilenumber' 
一 $mobilenumber,'content' => $content,'msgtype' => $msgtype))” 表 示 向 Web Service 传递 的 参数 。 


公 注意 : 在 进行 call0 方 法 中 的 参数 传递 时 ， 必 须 是 以 数组 的 形式 进行 传递 。 
9.8.4 ”单元 测试 
在 完成 短信 发 送 模块 的 设计 后 ， 为 保证 程序 正常 运行 ， 一 定 要 对 模块 进行 单元 测试 。 单 元 测试 在 


程序 开发 中 非常 重要 ， 只 有 通过 单元 测试 才能 发 现 模块 中 的 不 足 之 处 ， 才 能 及 时 地 弥补 程序 中 出 现 的 
错误 。 下 面 将 对 发 送 短信 模块 中 出 现 的 错误 进行 分 析 。 


在 短信 发 送 页 面 中 完成 信息 的 添加 之 后 , 单 击 “ 提 交 ” ET 
按钮 ， 其 运行 结果 如 图 9.41 所 示 。 EEC 放 


和 一 网 ee:wizroo VTsofy wo/seniz| 加 » 
根据 错误 信息 中 的 提示 ， 发 现 是 程序 中 没有 找到 。 记 ees/2o n/m Ee 


Fatal error: Class ’ soapclient’ not found in 


soapclient 这 个 类 ， 查 找到 send.php 文件 中 的 第 22 行 ， 发 Ee 


厂矿 厂 导 emt 


现 是 在 程序 中 没有 调用 nusoap.php 文件 , 从 而 导致 没有 找 
到 soapclient 类 。 对 该 错误 进行 修改 ， 将 nusoap.php 文件 图 941 发 送 短信 的 错误 运行 结果 
复制 到 程序 的 主 目录 下 进行 保存 ， 并 通过 inleude0 函 数 调用 nusoap/lib/nusoap.php 类 文件 ， 重 新 保存 程 
序 代码 ， 短 信 发 送 成 功 。 


9.9 连接 邮件 接口 模块 的 设计 


9.9.1 连接 邮件 接口 模块 概述 


在 本 项 目 中 邮件 收发 功能 是 通过 imap 电子 邮件 系统 函 
数 来 完成 的 ， 所 以 在 开发 程序 之 前 要 将 php.ini 文件 中 


Dynamic Extensions 模块 内 的 “;extension=php_imap.dll” 前 et 
的 “;” 去 掉 ， 然 后 重新 启动 Apache 服务 器 。 部 箱 : FREE 

连接 邮件 接口 模块 的 主要 功能 是 与 指定 的 POP3 服务 器 和 
建立 连接 并 实现 用 户 身 份 验 证 ， 其 运行 结果 如 图 9.42 所 示 。 [RR | a) 


全 注意 : 在 实现 邮件 收发 功能 之 前 首先 要 在 POP3 服务 器 
中 创建 一 个 新 邮箱 Tsoft@sina.com， 密 码 为 111， 
用 于 登录 到 服务 器 。 


图 9.42 ”邮件 服务 器 登录 页 面 


9.9.2 ”连接 邮件 接口 模块 技术 分 析 


在 连接 邮件 接口 模块 的 设计 中 ， 主 要 应 用 的 是 imap 电子 邮件 系统 函数 中 的 imap_open0 函 数 ， 通 
过 该 函数 实现 与 IMAP 服务 器 建立 连接 。imap_open0) 函 数 的 语法 如 下 : 
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resource imap_open(string mailbox, string username. string password [int options [int n_retries]]) 
imap_open0 函 数 的 参数 说 明 如 表 9.10 所 示 。 
表 9.10 imap_open() 函 数 的 参数 说 明 


参数 说 明 
mailbox 必 选 参数 。 服 务 器 地 址 
Username 必 选 参数 。 用 户 账 号 
password 必 选 参数 。 用 户 密码 
options 可 选 参数 。 该 参数 取 值 及 说 明 如 表 9.11 所 示 
int n retries 可 选 参数 。 试 图 与 IMAP 服务 器 建立 连接 的 最 大 连接 数 
表 9.11 options 参数 的 说 明 
参数 取 值 说 了 明 
OP READONLY 打开 链接 使 用 只 读 状 态 
OP _ ANONYMOUS 匿名 读 取 NNTP 服务 器 ， 不 使 用 newsrc 文件 
OP_HALFOPEN 只 与 IMAP 或 NNTP 服务 器 链接 ， 不 打开 邮箱 
CL_EXPUNGE 关闭 链接 时 自动 清除 邮箱 中 的 信件 


避 注意 : 使 用 imap_open() 函 数 连 接 邮件 服务 器 时 ， 在 Internet 和 局 域 网 上 连接 邮件 服务 器 有 所 不 
同 。 例 如 在 Intemet 上 应 用 imap_open() 函 数 连接 163 的 邮件 服务 器 ， 并 且 用 户 的 邮箱 地 址 
为 “mr***(@163.com”， 密 码 为 123456， 则 imap_open(O 函 数 的 格式 为 : 
$mbox=imap_open(" {pop3.163.com:110/pop3}", "mr***"."123456"):; 
如 果 用 户 在 局 域 网 上 建立 pop3 服务 器 ， 并 且 服务 器 的 卫 为 192.168.1.9， 在 POP3 服务 器 上 
建立 的 邮箱 为 mingrisoft@mingrisoft.com， 密 码 为 123456， 则 imap_open() 函 数 的 格式 为 : 
$mbox=imap_open("{192.168.1.9:110/pop3}","mingrisoft@mingrisoft.com","123456"); 


9.9.3 ”连接 邮件 接口 模块 的 实现 过 程 


连接 邮件 接口 模块 通过 mail.php 和 mail_user.php 两 个 文件 来 完成 。 首 先 通 过 mail.php 文件 创建 一 
个 表单 ,提交 服务 器 的 全 地 址 、 邮 箱 名 称 和 密码 到 mail_user.php 文件 中 ,然后 在 mail_user.php 文件 中 
应 用 imap_open0) 函 数 实现 登录 邮箱 的 验证 。mail userphp 文件 的 关键 代码 如 下 : 

例 程 11 代码 位 置 : 光盘 \TMWO9\qykxmail userphp 


<Iphp 
session start(): // 初 始 化 session 变 量 
Shostname="{".$_POST[hostname].":110/pop3}": // 获 取 局 域 网 服务 器 的 人 P 地 址 
Susername=$ POST[usermmame]: // 获 取 邮 箱 名 称 
Suserpwd=$_POST[userpwd]: // 获 取 邮 箱 密码 


0 if(!$Smbox=@imap_open("$hostname"."$usernmame"."$userpwd")){ /应 用 imap_open 函 数 实现 登录 邮箱 的 验证 
echo "<sctipt>alert(' 登 录 失 败 !):history.backO:</scripf": 
}else{ 
Session register("host"):; // 定 义 一 个 session 变 量 host 
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Session register("user"): /定义 一 个 session 变 量 user 
Session register("pwd"): /定义 一 个 session 变 量 pwd 
$_SESSION[host]=$hostname: /将 session 的 值 赋 给 变量 Shostname 
$_SESSION[user]=$username: // 将 session 的 值 赋 给 变量 Susername 
$_SESSION[pwd]=Suserpwd: // 将 session 的 值 赋 给 变量 $userpwd 

© imap_close($mbox): /关闭 由 imap_open0) 函 数 所 返回 的 连接 标识 

echo "<script>window.location.hre 人 >"indexs.php?lImbs= 登 录 ':</script>":; 
} 
?> 
< 代码 贴 十 


@ imap_open0 〇 : 实现 登录 邮箱 的 验证 ， 有 关 该 函数 的 详细 讲解 请 参考 连接 邮件 接口 技术 分 析 中 的 内 容 。 
@ imap_close0: 关闭 由 imap_open0 函 数 所 返回 的 连接 标识 。 


bool imap_close( resource imap_stream [.int flag] ) 


imap_stream 为 必 选 参数 ， 是 imap_open() 函 数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 。flag 为 可 选 参数 ， 如 果 
将 该 参数 设置 为 CL EXPUNGE， 则 在 断 开 与 服务 器 的 连接 同时 删除 所 有 带 有 连接 标记 的 邮件 。 


9.10 接收 邮件 模块 设计 


9.10.1 接收 邮件 模块 概述 


在 邮件 接收 模块 中 主要 实现 的 是 从 服务 器 中 读 取 邮件 的 内 容 ， 并 且 将 内 容 输出 到 页 面 中 。 在 接收 
邮件 模块 中 输出 邮件 的 内 容 ， 对 邮件 进行 分 页 显示 ， 而 且 可 以 对 指定 的 邮件 执行 删除 和 查看 操作 。 运 
行 结果 如 图 9.43 所 示 。 


| 快 信 V1.0 


发 件 人 发 位 时 间 

查找 种 pl FM rrz@sina com 2007-12-29 10:4 2.37 0 
发送 记 录 。。 可 ] 共有 邮件 1 封 每 页 旺 示 10 对 向 1 页 / 共 1 页 ) 1 
“刷新 


“退出 


图 9.43 接收 邮件 模块 的 运行 结果 
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9.10.2 ”接收 邮件 模块 技术 分 析 


在 接收 邮件 模块 中 主要 应 用 的 是 imap_checkO 函 数 、imap_searchO 函 数 和 imap_headerinfo0 函 数 ， 
其 中 通过 imap_check0 函 数 获取 连接 id， 并 返回 包含 当前 邮箱 信息 的 对 象 ， 包 括 邮 件数 目 ， 通 过 
imap_search() 函 数 搜寻 指定 标准 的 信件 ; 通过 imap_headerinfo0 函 数 获取 某 信件 的 标 头 信息 。 

imap_checkO 函 数 的 语法 格式 如 下 : 

object imap_check(resource imap_stream) 

参数 imap_stream 为 imap_open0 函 数 成 功 连接 上 邮件 服务 器 后 所 返回 的 连接 标识 。 该 函数 返回 对 
象 可 以 调用 的 属性 说 明 如 表 9.12 所 示 。 

表 9.12 imap_check() 函 数 返 回 对 象 可 调用 的 属性 说 明 


属 性 名 说 ”了 明 
Date 根据 RFC2822 时 间 格 式 所 返回 的 系统 当前 时 间 
Driver 返回 邮箱 所 使 用 协议 的 名 称 ， 包 括 POP3、IMAP 和 NNTP 
Mailbox 返回 邮箱 的 名 称 
Nmsgs 返回 邮箱 中 邮件 的 个 数 
Recent 返回 最 近 收 到 邮件 的 数目 


imap_search() 函 数 的 语法 格式 如 下 : 
array imap_search(resource imap_stream, string criterial, int options [,string charset]]) 
imap_search() 函 数 的 参数 说 明 如 表 9.13 所 示 。 

表 9.13 imap_search() 函 数 的 参数 说 明 


en 函数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 


criteria 必 选 参数 。 指 定 查询 的 条 件 
options 必 选 参数 。 为 SE_UID 值 
charset 可 选 参数 。 指 定 邮件 所 采用 的 字符 集 


imap_headerinfo0 函 数 是 imap_header0 函 数 的 别名 ，imap_headerinfo0 函 数 的 用 法 与 imap_header() 
函数 的 用 法 是 相同 的 ,这 里 只 对 imap_header() 函 数 进行 详细 的 介绍 。imap_header() 函 数 的 语法 格式 
如 下 : 

object imap_header(int imap_stream.int msg_number.int [fromlength].int [subjectlength].string [defaulthost]): 

imap_header(O) 函 数 的 参数 说 明 如 表 9.14 所 示 ， 该 函数 返回 对 象 可 以 调用 的 属性 说 明 如 表 9.15 
所 示 。 
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表 9.14 imap_header() 函 数 的 参数 说 明 


参 数 说 了 明 
imap_stream 必 选 参数 。imap_open0 函 数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 
Imsg_number 必 选 参数 。 邮 件 号 
fromlength 必 选 参数 。 指 定 发 件 人 地 址 长 度 
subjectlength 必 选 参数 。 指 定 主题 长 度 
defaulthost 必 选 参数 。 指 定 默认 主机 号 

表 9.15 imap_header() 函 数 返 回 对 象 可 调用 的 属性 说 明 

属 性 值 说 了 明 
Answered 判断 邮件 是 否 已 经 回复 ， 是 则 返回 A， 否 则 返回 空 
bcc 返回 与 Bec 有 关 的 的 对 象 数 组 
bccaddress 返回 密 送 人 地 址 
ce 返回 与 Cc 有 关 的 的 对 象 数组 
ccaddress, 返回 抄 送 人 地 址 
date 邮件 被 创建 的 时 间 
Date 与 date 用 法 相同 
Deleted 判断 邮件 是 否 被 打上 删除 标记 ， 是 则 返回 D， 否 则 返回 空 
from 返回 与 发 件 人 信息 有 关 的 对 象 数组 
fromaddress 发 件 人 地 址 
Msgno 返回 邮件 号 
Recent 如 果 是 最 近 的 邮件 并 且 已 经 查看 了 则 返回 R， 否 则 返回 空 
Size 返回 邮件 的 大 小 
Subject 返回 邮件 主题 
Subject 与 Subject 用 法 相同 
to 返回 与 收 件 人 有 关 的 对 象 数 组 
toaddress 返回 收 件 人 地 址 
Unseen 如 果 邮 件 已 经 被 查看 则 返回 U， 否 则 返回 空 值 


在 接收 邮件 模块 中 首先 通过 imap_open0 函 数 来 登录 到 服务 器 ， 然 后 应 用 imap_checkO 函 数 获取 
imap_open() 函 数 成 功 连接 后 返回 的 连接 标识 ， 根 据 获 取 到 的 连接 标识 ， 通 过 imap_search(O) 函 数 搜寻 指 
定 标准 的 信件 , 应 用 imap_headerinfo0 函 数 获取 某 信件 的 标 头 信息 ,最 后 通过 分 页 技术 将 获取 到 的 邮件 
信息 进行 输出 。 程 序 的 关键 代码 如 下 : 

连接 邮件 服务 器 ， 创 建 一 个 分 页 技术 ， 输 出 邮件 中 信息 的 关键 代码 如 下 : 


例 程 12 ”代码 位 置 : 光盘 \TMWO9\qykeelookmailLphp 


< ?php session start(): /初始 化 session 变 量 
S$hostname=$_ SESSION[host]: // 获 取 表单 中 提交 的 服务 器 IP 
$usemame=$_ SESSION[user]: /获取 用 户 名 
$userpwd=$_ SESSION[pwd]: /获取 密码 


/检测 呈 站 有 7 豚 iD3a0 Open 国 数 ， 实现 登录 服务 器 的 验证 村 字音 人 
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if(!$mbox=@imap_open("$hosthame","$username"."$userpwd")){ 


echo "<script>alert( 登 录 超时 ， 请 重新 登录 !"):history.back(:</script>"; 


exit: 


} 


?> 

<?php 
Scheck = imap_check($mbox); 
S$sum=$check->Nmsgs: 
Print r(imap_search($mbox,"SEEN")); 
这 $sum<=0){ 

?> 


/省 略 了 部 分 代码 


// 获 取 imap_open 函 数 返 回 的 连接 标识 
// 将 连接 标识 赋 给 变量 

// 根 据 连 接 标 识 查找 指定 标准 的 信件 
// 判 断 当 返回 值 为 空 时 输出 如 下 内 容 


<tr><td height="25" colspan="5" align="center"> 暂 无 邮件 </td></tr> 


<?php 
}else{ 


* 编 写 一 个 分 页 技术 ， 用 于 分 页 显示 邮件 信息 */ 


// 判 断 当 返 回 值 不 为 空 时 执行 下 面 的 内 容 


if($_GET[page] 一 "" || is_numeric($_GET[page]=false)){ /判断 当 变 量 Spage 的 值 不 存在 或 为 空 时 


S$page=1; 
}else{ 
$page=$_GET[page]: 


S$pagesize=10; 
if($sum%$pagesize——0){ 

Stotalpage=$sum/$pagesize; 
}else{ 


Stotalpage=ceil($sum/$pagesize): 


} 

$frompage=($page-1)*$pagesize+l: 

$topage=$frompage+S$pagesize: 

这 ($sum-S$topage)<0){ 
$topage=$sum+1: 


} 
for($i=$frompage:$i<$topage:Si+H){ 


S$obj=imap_headerinfo($mbox.$i):; 


ER */ 


?> 


9.10.3 ”查看 邮件 的 实现 过 程 


/定义 变量 $page 的 值 为 1 

// 如 果 不 为 空 ， 则 定义 $page 值 为 当前 变量 值 
/定义 变量 Spagesize 的 值 为 10 

// 对 邮件 信息 进行 分 页 ， 每 页 显示 10 条 

// 定 义 变量 $totalpage 输 出 共有 几 页 信息 

// 对 分 页 进行 取 整 


// 定 义 $Sfrompage 每 页 的 第 一 条 记录 
/定义 Stoppage 表 示 每 页 的 最 后 一 条 记录 


/应 用 for 循 环 语句 实现 页 面 跳 转 
/获取 邮件 的 标 头 信息 


查看 邮件 的 功能 主要 通过 imap_header0 函 数 、imap_fetchbody0 函 数 和 imap_fetchstructureO 函 数 实 


现 ， 其 运行 结果 如 图 9.44 所 示 。 


其 中 imap_header0 函 数 已 经 在 技术 分 析 中 详细 讲解 过 ， 这 里 主要 对 imap_fetchbody0O 函 数 和 


imap_fetchstructure(O) 函 数 进行 详细 的 讲解 。 


imap_fetchbody0 函 数 用 于 获取 邮件 指定 部 分 的 内 容 ， 其 语法 格式 如 下 : 
string imap fetchbody(resource imap_stream.int msg number.string part_ number [.int options]) 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


“区 到 # 7 是 区 “| “1 欢迎: 欢迎 您 得 录 邮件 收发 管理 系 纺 。 
| ER 


时 间 : 2007-12-29 10:4:53 


9.44 查看 邮件 的 运行 结果 
imap_fetchbody0 函 数 的 参数 说 明 如 表 9.16 所 示 。 
表 9.16 imap_fetchbody() 函 数 的 参数 说 明 


必 选 参数 。imap_open0 函 数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 


必 选 参数 。 邮 件 号 
必 选 参数 。 指 定 邮件 部 分 号 


imap_fetchstructure() 函 数 用 于 获取 邮件 的 结构 。 其 语法 格式 如 下 : 
object imap_fetchstructure(resource imap_stream.int msg_Dumber[.int options]) 
inmap_fetchstructureO 函 数 的 参数 说 明 如 表 9.17 所 示 。 

表 9.17 imap_fetchstructure() 函 数 的 参数 说 明 


imap_stream 必 选 参数 。imap_open0 函 数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 
msg_number 必 选 参数 。 邮 件 号 
options 可 选 参数 


通过 上 述 3 个 函数 的 合理 应 用 ， 就 可 以 轻松 地 完成 邮件 的 查看 功能 。 首 先 与 邮件 服务 器 建立 连接 ， 
并 根据 连接 标识 和 ID 返回 邮件 标 头 信息 ， 代 码 如 下 : 
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例 程 13 ”代码 位 置 ， 光盘 \TMNO9Aqylealookmailinfo php 


<?php session start(): // 初 始 化 session 变 量 
Shostname=$_SESSION[host]: /获取 表单 中 提交 的 服务 器 
Susemame=$_SESSION[user]: // 获 取 用 户 名 
Suserpwd=$_SESSION[pwd]: /获取 密码 


/应 用 ipam _ open 函数 ， 实 现 登录 服务 器 的 验证 
if(!'$mbox=@imap_open("$hosthname","$username","$userpwd")){ 
echo "<script>alert( 登 录 超时 ， 请 重新 登录 !"):history.back(:</script>"; 


exit; 
} 
$id=$_GET[id]: /获取 ID 的 值 
$obj=(@imap_header($mbox,S$id): // 根 据 imap_open 函 数 返 回 的 连接 标识 和 I 了 D 的 值 来 获取 邮件 标 头 信息 
?> 


然后 根据 获取 的 邮件 标 头 信息 ， 输 出 邮件 的 发 送 时 间 、 发 件 人 、 收 件 人 和 邮件 的 主题 ， 代 码 如 下 : 
例 程 14 ”代码 位 置 : 光盘 \TM\09\qykx\lookmailinfo.php 


<?php 

/获取 邮件 发 送 的 时 间 

Sarray=(Ogetdate(strtotime($obj->date)): 

echo 
Sarray[year]."-".$array[mon]."-".$array[mday]."é&nbsp:".$array[hours].":".$array[minutes].":".$array[seconds]; 

?> 
<?php 。 echo $obj->fromaddress: /获取 发 件 人 ?> 
<?php 。 echo $obj->toaddress: /获取 收 件 人 ?> 


<?php /获取 邮件 的 主题 
if(strtolower((@substr($obj->Subject,0.10))=—=strtolower("=?gb2312?B")) 
echo base64_decode(substr($obj->Subject.11.,(strlen($obj->Subject)-13))): 
else 
echo $obj->Subject: 
?> 


< 人 代码 贴 二 


@ getdate0: 获取 日 期 和 时 间 信 息 。 返 回 一 个 根据 timestamp 得 出 的 包含 有 日 期 信息 的 结合 数组 。 如 果 没 有 给 出 时 


间 狼 则 认为 是 当前 本 地 时 间 。 


@ strtolower0: 将 字符 转换 为 小 写字 母 。 
上 @ base64 decode(0: 对 使 用 MIME base64 编码 的 数据 进行 解码 。substr0: 从 指定 的 字符 串 中 按照 指定 的 位 置 截 取 


一 定 长 度 的 字符 。strlen0 获 取 指 定 字符 串 的 长 度 。 


最 后 获取 邮件 中 附件 的 名 称 和 邮件 的 内 容 ， 并 且 设 置 下 载 附件 的 超 链接 ， 代 码 如 下 : 
例 程 15 ”代码 位 置 : 光盘 \TM\W9\Wqykxlookmailinfo php 


<?php 
/这 于 素 妆 守 率 于 认 六 认 六 率 六 率 闪 率 不 率 让 率 站 率 相 让 相 补 守卫 有 了 付 件 :各 和 二 本 本 二 本寺 本 中 本 本 本 本本 中环 可 中 本 可 呈 本 可 呈 水 可 本 可 字 守 于 束 让 / 

$structure= (@imap_fetchstructure($Smbox.$id): // 获 取 邮 件 的 结构 ， 输 出 附件 的 名 称 
S$array=$structure->parts; 


if(($array[1]->dparameters[0]->value)!=""){ 
Sfilename=$array[1]->dparameters[0]->value: 
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Jelse{ 
Sfilename=$array[1]->description:; 
} 
if(strtolower(substr($filename,0.10))=—=strtolower("=?gb2312?B")) 
echo base64 decode(substr($filename,11.(strlen($filename)-13))): /1/ 从 信息 中 截取 附件 的 名 称 
else 
echo $filename; // 输 出 附件 名 称 


?> 
<!-- 设 置 下 载 附件 的 超 链接 -> 
<td width="61"><a href="down.php?id=<?php echo $id:?>" class="a1">( 下 载 附件 )</a></td> 


区 Ss 

<?php 

/机 本 中 下 了 国人 办] 从 这 家 守 / 
include_once("function .php"): /调用 该 文件 实现 特殊 字符 替换 


/应 用 imap_fetchbody 函 数 获取 邮件 指定 部 分 的 内 容 
这 unhtml(imap_base64(@imap_fetchbody($mbox.Sid.1))) 一 ""){ 
echo unhtml(@imap fetchbody($mbox,$id,1)): 
}else{ 
echo unhtml(imap_base64(imap fetchbody($mbox,$id.,1))): 
有 


?> 


0.4 删除 邮件 的 实现 过 程 


删除 邮件 功能 的 实现 应 用 到 imap_open0 函 数 、imap_delete0 函 数 、imap_close0 函 数 和 imap_expungeO 


函数 。 其 中 主要 应 用 的 是 imap_delete0 函 数 ， 为 指定 的 邮件 打上 删除 标记 ,应 用 imap_expunge0 函 数 删 
除 所 有 带 有 删除 标记 的 邮件 。 


imap_delete() 函 数 为 指定 的 邮件 打上 删除 标记 ， 然 后 应 用 imap_expungeO 函 数 或 者 imap_close() 函 


数 实现 邮件 的 物理 删除 。 语 法 格式 如 下 : 


bool imap_delete(int imap_stream.int msg_number [,int options]) 
imap_delete(O) 函 数 的 参数 说 明 如 表 9.18 所 示 。 
表 9.18 imap_delete() 函 数 的 参数 说 明 
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参 。 数 说 了 明 

imap stream | 必 选 参数 。imap_open0 函 数 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 
msg_number | 必 选 参数 。 邮 件 号 

options 可 选 参数 


imap_expunge() 函 数 用 于 删除 所 有 带 有 删除 标记 的 邮件 。 其 语法 格式 如 下 : 


bool imap_expunge(resource imap stream) 


参数 imap_stream 为 imap_open0 函 数 成 功 连接 上 服务 器 后 的 返回 值 。 
删除 邮件 功能 的 实现 主要 通过 lookmail.php 和 delmail.php 文件 来 完成 。 首 先 在 lookmail.php 文件 
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吕 


好 


Ph 创建 一 个 表单 ， 设 置 复 选 框 ， 用 来 选择 要 删除 的 邮件 ,创建 “删除 ”按钮 ， 将 复 选 框 的 值 提交 到 
delmail.php 页 中 ; 在 delmail.php 页 中 根据 表单 中 提交 的 值 ， 应 用 imap_delete0 函 数 为 指定 的 邮件 打上 


| 除 标记 ， 最 后 应 用 imap_expunge0) 函 数 删除 所 有 带 有 删除 标记 的 邮件 。 其 中 delmail.php 文件 的 关键 
代码 如 下 : 
例 程 16 ”代码 位 置 : 光盘 \TM\09\qykx\delmail.php 
<?php session start(): // 初 始 化 session 变 量 
Shostname=$_SESSION[host]: /获取 表单 中 提交 的 服务 器 下 
S$usemame=$_ SESSION[user]: // 获 取 用 户 名 
$userpwd=$_SESSION[pwd]: /获取 密码 


/应 用 ipanm_ open 函数 ， 实 现 登录 服务 器 的 验证 
if(!$mbox=@imap_open("$hosthame","$username","$userpwd")){ 
echo "<script>alert( 登 录 超时 ， 请 重新 登录 !"):history.backO:</script>"; 
exit; 


} 


$i=0; // 定 义 变量 $i=0 
while(list(S$name,$value)=each($ POST)){ // 应 用 list 函 数 获取 表单 中 提交 的 值 ， 应 用 while 语 句 循环 输出 list 函 
数 返 回 的 值 
ifis_numeric($value) 一 tmue){ // 判 断 $Svalue 的 值 是 否 为 真 
$i+=$value: 
if(!@imap_delete($mbox,S$value)){ // 根 据 $value 的 值 和 连接 标识 为 指定 的 邮件 打上 删除 标记 
echo "<script>alert( 删 除 失败 !):history.backO:</script>": 
exit; 
} 
} 
} 
if($i=—0){ // 判 断 当 $i=0 时 执行 下 面 的 内 容 
echo "<script>alert(' 请 选择 要 删除 的 邮件 !"):history.backO;</script>"; 
imap_close($mbox); // 关 闭 由 imap_open0 函 数 所 返回 的 连接 标识 
exit; 
j}else{ 
imap_expunge($mbox): // 判 断 $i 不 等 于 0 时 执行 imap_expunge， 删 除 打上 删除 标记 的 邮件 
imap_close($mbox): // 关 闭 由 imap_open0 函 数 所 返回 的 连接 标识 


echo "<script>window.location.href='indexs.php?lmbs= 删 除 '</script>"; 
} 


ye 


9.10.5 下 载 附 件 的 实现 过 程 


在 下 载 附 件 模块 中 同样 是 应 用 imap_open0) 函 数 、imap_header0 函 数 、imap fetchbody0 函 数 和 


imap_fetchstructure0 函 数 来 实现 。 有 关 这 些 函 数 的 详细 讲解 请 参考 前 面 的 内 容 ， 这 里 不 再 袭 述 。 下 载 附 


件 模块 的 运行 结果 如 图 9.45 所 示 。 


下 载 附件 功能 的 实现 从 查看 邮件 页 面 中 的 下 载 附件 超 链 接 开 始 ， 在 超 链接 中 设置 一 个 变量 id， 作 


为 指定 附件 的 唯一 标识 ， 该 超 链接 链接 到 down.php 文件 ， 在 该 文件 中 完成 对 指定 附件 的 下 载 。 关 键 代 


码 如 下 : 
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对 骸 区 | 欢迎 : 欢迎 燃 营 录 邮件 收发 管理 系统 。 


“查找 邮件 。 您 想 打开 或 保存 此 文件 吧 ? 
“发 送 记录 。 严 和 名称， 新建 文本 文档 (3). tzt 


“退出 


例 程 17 ”代码 位 置 : 光盘 \TMW9vqykxvdown.php 


<?php session start(): // 初 始 化 session 变 量 
$hostname=$_SESSION[host]: // 获 取 表单 中 提交 的 服务 器 他 
Susername=$_SESSION[user]: // 获 取 用 户 名 
Suserpwd=$_SESSION[pwd]: // 获 取 密 码 


// 应 用 ipam_open 函 数 ， 实 现 登 录 服务 器 的 验证 
if(!Smbox=@imap_open("$hostname","$username","$userpwd")){ 
echo "<script>alert(' 登 录 超时 ， 请 重新 登录 !"):history.backO:</script>"; 
exit; 


} 


S$structure= imap_fetchstructure(Smbox.$id): // 根 据 提交 的 变量 ID 获 取 指定 邮件 的 结构 
S$array=$structure->parts; // 将 邮件 结果 的 信息 赋 给 数组 Sarray 
if(($array[1]->dparameters[0]->value)!=""){ /判断 数组 中 指定 的 数据 是 否 为 空 
Sfilename=$array[1]->dparameters[0]->value: // 将 文件 的 名 称 赋 给 变量 $filename 
}else{ 
Sfilename=$array[1]->description: 


} 

if(strtolower(substr($filename.0.10))==strtolower("=?gb2312?B")) 
$filename=base64 decode(substr($filename.11,(strlen($filename)-13))): 

header("Content-type:application/octet-stream"): 

header("Accept-ranges:bytes"): 

header("Accept-length:100"): 

header("Content-Disposition:attachment:filename=".$filename.""): 


// 对 文件 的 名 称 进行 截取 
// 对 文件 的 名 称 进行 截取 ， 并 进行 解码 


Stext=imap_fetchbody($mbox.$id.2): // 根 据 提供 的 ID， 获 取 邮 件 第 2 部 分 的 内 容 
echo imap base64(Stext): // 对 base64 编 码 的 文本 进行 解码 
imap_close($mbox): // 关 闭 由 imap_open0 函 数 所 返回 的 连接 标识 
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exit; 
?> 
< 代码 贴 二 
@ Qimap open0: 与 IMAP 服务 器 建立 连接 。 
@ imap _fetchstructure0: 获取 邮件 的 结构 。 语 法 如 下 : 


object imap fetchstructure(resource imap_stream.int msg_number[.int options]) 


imap_stream: 必 选 参数 ， 创 建 与 服务 器 成 功 建立 连接 后 所 返回 的 连接 标识 ， 参 数 msg_number: 指 邮件 号 。 
上 @ imap_fetchbody0: 获取 指定 邮件 的 指定 部 分 的 内 容 。 


9.11 发 送 邮 件 模块 设计 


9.11.1 ”发送 邮件 模块 概述 


发 送 邮 件 模块 主要 用 于 邮件 的 发 送 ， 并 且 可 以 实现 邮件 群发 和 添加 附件 的 功能 。 在 实现 邮件 群发 
的 过 程 中 ， 添 加 收 件 人 的 邮箱 地 址 时 必须 使 用 * 进 行 分 隔 ， 否 则 将 不 能 实现 。 发 送 邮件 模块 的 运行 结果 
如 图 9.46 所 示 。 


到 家 
主题: 网 FE 
rt | 


ET EE [ 


Fa [二 | 


图 946 发 送 邮 件 模块 的 运行 结果 
9.11.2 发送 邮 件 模 块 技术 分 析 


邮件 发 送 功能 的 实现 主要 通过 imap_mail compose0O 函 数 和 imap_mail0 函 数 来 完成 。 下 面 详细 讲解 


这 两 个 函数 的 使 用 方法 。 
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imap_mail compose() 函 数 用 于 创建 一 个 MIME 邮件。 语法 格式 如 下 : 

string imap_mail compose(array envelope. array body) 

envelope 为 必 选 参数 , 由 与 邮件 地 址 有 关 的 首部 信息 组 成 , 包括 From、Reply_To、Cc、Bcc、Subject 
等 项 ;body 为 必 选 参数 ， 由 具体 的 邮件 以 及 与 其 格式 有 关 的 各 种 属性 组 成 。 

imap_mail0 函 数 用 于 发 送 邮件 ， 语 法 格式 如 下 : 

bool imap_ mail(string to, string subject string message [,string additional headers [, string cc [.string bcc[.string rpath]]]) 

imap_mail0 函 数 的 参数 说 明 如 表 9.19 所 示 。 

表 9.19 imap_mail() 函 数 的 参数 说 明 


参数 说 了 明 
to 必 选 参数 。 收 件 人 地 址 
Subject 必 选 参数 。 邮 件 主题 


messapge 必 选 参数 。 邮 件 内 容 
additional_ headers 邮件 额外 首部 信息 
抄 送 人 地 址 


密 送 人 地 址 
用 于 设置 Return-Path 首 前 


9.11.3 ”发 送 邮件 的 实现 过 程 


国 发送 邮 件 使 用 的 数据 表 : tb_mail 

发 送 邮 件 的 实现 过 程 主要 应 用 sendmail.php 文件 和 mail_send.php 文件 。 首 先 通过 sendmail.php 文 
件 创建 一 个 form 表单 ， 用 于 提交 邮件 的 发 件 人 、 收 件 人 、 主 题 、 附 件 和 内 容 信息 ， 将 该 信息 提交 到 
mail_send.php 文件 中 进行 处 理 。 其 中 使 用 的 form 表单 的 属性 如 表 9.20 所 示 。 


表 9.20 发送 邮件 模块 中 使 用 的 重要 表单 元 素 


名 称 | 元素 类 型 重要 属性 含义 
ethod "peat" eetiom “mail_send.php” enctype= "multipart/form-data" 表单 
onSubmit="return chkinput(this) 
class="inputcss" id="fromuser" style="background-color:#e8f4ff " 
i GnMobseOver “tsstyie, backenyungd elo HE 发 件 人 
onMouseOut="this.style.backeroundColor='#e8f4ff" value="<?php echo 
$_SESSION[user]:?>" size="50" 
class="inputcss" id="touser" style="backeround-color:#e8f4ff " 

touser textarea onMouseOver= "this.style.backgroundColor#ffffff” 收 件 人 

onMouseOut="this.style.backgroundColor=#e8f4ff"> 
class="inputcss" id="subject" style="background-color:#e8f4ff " 
subject text onMouseOver= "this.style.backgroundColor#ffffff” 主题 
onMouseOut="this.style.backeroundColor='#e8f4 人 ff" size="50" 
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续 表 


名 称 | 元素 类 型 重要 属性 含义 


upfile file class="inputcss" id="upfile" size="50" 附件 


IOWS="15" class="inputcss" id="mailbody" style="BORDER-RIGHT: 
#aaaaaa 1px solid; BORDER-TOP: #aaaaaa 1px solid: OVERFLOW-Y: 
scroll; BORDER-LEFT: #aaaaaa 1px solid: WIDTH: 320px: 
textarea 内 容 
SCROLLBAR-SHADOW-COLOR: #556688: 

SCROLLBAR-ARROW-COLOR: #556688: BORDER-BOTTOM: #aaaaaa 


1px solid; SCROLLBAR-BASE-COLOR: #1C92E2; HEIGHT: 120px" 


mailbody 


submit submit class="buttoncss" id="submit" value=" 发 送 " 


提交 表单 


全 注意 : 在 添加 收 件 人 地 址 时 ， 可 以 直接 将 地 址 输入 到 文本 框 中 ; 也 可 以 从 右 侧 的 下 拉 菜 单 中 选择 


对 应 的 客户 或 者 同事 ， 单 击 客户 或 者 同事 的 名 称 来 添加 收 件 人 ， 该 技术 的 实现 方法 已 经 在 
系统 信息 管理 模块 技术 分 析 中 详细 地 讲解 过 ， 这 里 不 再 丙 述 。 如 果 在 添加 收 件 人 的 邮箱 地 
址 时 以 “*” 来 进行 分 隔 ， 那 么 就 可 以 实现 邮件 群发 的 功能 。 


在 mail_send.php 文件 中 对 表单 中 提交 的 数据 进行 处 理 , 从 而 应 用 imap_mail0 函 数 实现 邮件 发 送 的 


功能 。 关 键 代码 如 下 : 


例 程 18 ”代码 位 置 : 光盘 \TMW9\Wqykxmail send.php 
<2php 


if($_POST[submit]!=""){ // 判 断 提交 的 值 是 否 为 空 
Ssubject=$_POST[subject]: // 获 取 邮 件 主题 
Smailbody=$_POST[mailbody]: /获取 邮件 内 容 
Senvelope["from"]=$_POST[fromuser]: /获取 发 件 人 
Spartl["type"] = TYPEMULTIPART: 

$partl["subtype"] = "mixed"; 

Spart2["type"] = TYPETEXT: 

$part2["subtype"] = "plain"; 

$part2["encoding"] = ENCBINARY-: 

S$part2["contents.data"] = "$mailbody\n\n\n\t"; 
$filename =$_FILES[upfile][mame']: // 获 取 上 传 文件 的 真实 名 称 
if($filename!=""){ /判断 上 传 文件 是 否 为 空 
S$file=$ FILES[upfile]['tmp_name]: /获取 上 传 文件 的 临时 名 称 
Sfp = @fopen(S$file, "r"): /以 写 的 方式 打开 该 文件 
S$contents = (@fread($fp. Ofilesize($file)): // 读 取 该 文件 
@felose($fp): // 关 闭 文件 

这 $_FILES[upfile][type]){ /获取 上 传 文件 的 类 型 

SmimeType = $_FILES["upfile']['type']: // 将 文件 的 类 型 赋 给 变量 SmimeType 


jelse{ 

SmimeType ="application/unknown"; 
) 
Spart3["type"] = TYPEAPPLICATION: 
Spart3["encoding"] = ENCBINARY: 
Spart3["subtype"] = $mimeType: 
Spart3["description"] = $filename; 
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$part3["contents.data"] = $contents: 


} 
Sbody[1] = Spartl: 
Sbody[2] = Spart2: 
if($filename!=""){ // 判 断 上 传 文件 不 为 空 
Sbody[3] = Spart3: /| 执行 Spart3 


} 
$message=imap_mail compose($envelope, $body); // 创 建 一 个 MIME 上 邮件 
list($msgheader,$msgbody)=split("\rinrn",$Smessage,2); // 将 文件 中 的 数据 以 "rin" 为 分 制 符 ， 分 割 到 数组 中 


@ Sdata=trim($_POST[touser]): /去 除 收 件 人 字符 串 中 的 空格 
S$datas=explode("*".S$data): // 以 * 进 行 分 隔 ， 返 回 一 个 数组 
$mail date=date("Y-m-d H:i:s"): /获取 时 间 
© S$ip=getenv(REMOTE ADDR'): // 获 取 下 地 址 
while(list(Sname,$value)=each($datas)){ // 执 行 list 函 数 读 取 数 组 中 的 数据 ， 并 且 应 用 while 语 句 进行 循环 输出 
Ssendes=imap_mail($value,Ssubject,Smsgbody,Smsgheader); /执行 imap_mail0 函 数 实现 邮件 的 发 送 
include_once("conn/conn.php"): // 连 接 数 据 库 ， 将 邮件 的 发 送信 息 存 储 到 数据 库 中 


$sql="insert into 


,".$mail date.™)"; 
Srs=new com("adodb.recordset"): 
S$rs->open($sql,$conn,3,1); 
} 
if($sendes—true){ 
echo "<script>alert(' 邮 件 发 送 成 功 !"):history.backQ;</script>"; 
}else{ 
echo "<script>alert( 邮 件 发 送 失 败 !"):history.backO:</script>"; 
} 
?> 
Ah 代码 贴 二 
@ fopen: 打开 某 文件 ， 并 返回 该 文件 的 标识 指针 .。 
@ fread: 从 文件 指针 所 指 文件 中 读 取 指定 长 度 的 数据 。 
图 fclose: 关闭 指定 文件 标识 指针 所 指 的 文件 。 
@ tim0: 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 。 
@ getenv0: 获取 环境 变量 的 值 。 


9.11.4 查看 邮件 记录 的 实现 过 程 


国 ”查看 邮件 记录 使 用 的 数据 表 : tb_mail 

查看 邮件 记录 实现 的 是 对 发 送 邮件 的 记录 进行 查看 ， 主 要 就 是 读 取 数据 库 中 存储 的 有 关 邮 件 发 送 
的 记录 信息 。 运 行 结果 如 图 9.47 所 示 。 

查看 邮件 记录 功能 的 实现 主要 应 用 的 sendmail_ selectphp 文件 , 在 该 文件 中 首先 判断 用 户 是 否 登录 
邮件 服务 器 ， 如 果 已 经 登录 则 可 以 查看 数据 库存 储 的 有 关 邮 件 发 送 记录 的 数据 。 首 先 连接 数据 库 ， 然 
后 通过 PHP 的 预定 义 类 com 读 取 数据 库 中 的 数据 ， 通 过 while 循环 语句 输出 读 取 到 的 数据 ， 并 且 应 用 
分 页 技术 对 数据 信息 进行 分 页 显示 。 程 序 的 关键 代码 如 下 : 
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,发 > 条 EE 


-Nt 箱 邮件 二 是 发 件 人 发 1 时 间 路 件 人 
Mt 万 ba in com TT in eon 
-发送 记录 克 程序 测试 rrrzesina com 2007.1229 。。 rrrzesins com 


| tpttzg 二 页 显示 10 条 第 1 页 / 共 ! 页 1 


图 9.47 查看 邮件 记录 页 面 的 运行 结果 
例 程 19 代码 位 置 : 光盘 \TM\09\qykx\sendmail select.php 


<?php session start(): // 初 始 化 session 变 量 
$hostname=$_ SESSION[host]: /获取 表单 中 提交 的 服务 器 IP 
Susemame=$_SESSION[user]: /获取 用 户 名 

$userpwd=$_ SESSION[pwd]; /获取 密码 


// 应 用 ipam_open 函 数 ， 实 现 登录 服务 器 的 验证 
if(!$Smbox=(@imap_open("$hostname","$username"."$userpwd")){ 
echo "<script>alert(' 登 录 超时 ， 请 重新 登录 !"):history.backO:</script>"; 


exit; 
?> 
<?php 。 include("conn/conn .php"): // 连 接 数 据 库 
$sql="select * from tb_mail": // 创 建 查询 语句 ， 读 取 数 据 库 中 邮件 发 送 记 录 数 据 
Srs=new com("adodb.recordset"); // 应 用 PHP 预 定义 类 COM 
S$rs->open($sql.$conn.3.1): // 执 行 查询 语句 
/这 六 六 宗 闽 半 站 来 半 半 半 宗 闪 闫 ] 分 页 技术 的 实现 做 准备 功能 **** 认 下 于 下 站 守 杰 闪闪 / 
Srs->pagesize=10: /定义 变量 用 于 分 页 技术 的 实现 
/判断 当前 变量 的 值 是 否 存在 
if((trim(intval($_GET[page]))=—"")ll(intval($_GET[page])>$rs->pagecount)ll(intval($_GET[page])<=0){ 
Spage=1: /如 果 当 前 变量 不 存在 值 ， 则 定义 变量 $page 的 值 为 1 
}else{ 
Spage=intval($_GET[page]): } // 如 果 当 前 变量 存在 值 ， 则 获取 该 值 
if($rs->eof | $rs->bof){ 
?> 


<tr><td height="20" colspan="5" bgcolor="#FFFFFF"><div align="center"> 没 有 信息 ! </div></td></tr> 
<?php yelse{ 


Sres->absolutepage=$page: 

Smypagesize=$rs->pagesize: 

/于 六 率 六 六 六 闪 妆 站 六 宁 站 六 六 站 六 站 弟 亲 丰 六 站 亲 六 六 亲 亲 站 亲 亲 站 站 站 亲 站 六 六 站 训 亲 站 站 站 站 站 站 站 闲 让 站 亲 / 

while(!S$rs->eof &é& $mypagesize>0){ // 应 用 while 循 环 语句 输出 数据 库 中 数据 
?> 


<tr><td width="20"><input type="checkbox" name="check[]" value="<?php S$fields=S$rs->fields(mail id):echo 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


Sfields->value:?>" /></td> <td width="132" height="25"><div align="center"> 
<?php $fields=S$rs->fields(mail title):echo $fields->value:?></div></td> 
<td><div align="center"> 
<?php $fields=S$rs->fields(mail formuser):echo $fields->value:?></div></td> 
<td><div align="center"> 
<?php $fields=$rs->fields(mail_ date):echo $fields->value:?></div></td> 
<td><div align="center"> 
<?php $fields=S$rs->fields(mail touser):echo S$fields->value:?></div></td> 
</> 
<?php 
S$mypagesize--: 
Srs->movenext: // 将 查询 语句 的 指针 指向 下 一 条 语句 
和 
?> 
全 注意 : 在 应 用 $Is->open 语句 执行 查询 操作 时 ， 一 次 查询 执行 完成 后 ， 一 定 要 应 用 Srs->movenext 
语句 将 查询 的 指针 指向 下 一 条 记录 ， 否 则 查询 的 指针 不 会 自动 下 移 ， 执 行 下 一 条 查询 语句 ， 
将 一 直 停留 在 当前 位 置 ， 从 而 导致 只 能 输出 一 条 记录 。 


9.12 开发 技巧 与 难点 分 析 


9.12.1 PHP 与 Access 数据 库 的 连接 


在 本 项 目 中 使 用 的 是 Access 数据 库 ,在 建立 PHP 与 Acess 数 据 库 的 连接 时 使 用 的 是 微软 的 ADODB 
数据 库 驱 动 。 
微软 的 ADODB 数据 库 驱 动 即 ActiveX Data Objects (ADO) ， 是 Microsoft 开放 数据 库 应 用 程序 的 
数据 库 访 问 技术 。 它 被 用 来 设计 同 新 的 数据 访问 层 OLE DB Provider 一 起 协同 工作 ， 提 供 通 用 数据 访 
间 (Universal Date Access) 。OLE DB 是 一 个 低层 的 数据 访问 接口 ， 用 它 可 以 访问 各 种 数据 源 ， 包 括 
传统 的 关系 型 数据 库 ， 以 及 电子 邮件 系统 及 自 定义 的 商业 对 象 。ADO 技术 大 大 简化 OLE DB 的 操作 ， 
因为 ADO 封装 OLE DB 程序 中 使 用 的 大 量 COM 接口 ， 所 以 ADO 是 一 种 高 层 的 访问 技术 。 
ADO 技术 基于 通用 对 象 模型 (COM) ， 它 提供 多 种 语言 的 访问 技术 。PHP 是 通过 预先 定义 类 com 
来 使 用 ADO 方法 操纵 Access 数据 库 的。 该 类 详细 说 明 如 下 : 
string com::com( string module name [, string server name [, int codepage]]) 
其 参数 说 明 如 表 9.21 所 示 。 
表 9.21 预先 定义 类 com 的 参数 说 明 
参数 说 了 明 
module name | 被 请 求 组 件 的 名 字 或 class-id 
SeIVer name | DCOM 服务 器 的 名 字 


指定 用 于 将 PHP 字符 串 转换 成 UNICODE 字符 串 的 代码 页 ， 反 之 亦 然 。 该 参数 的 取 值 有 CP_ACP、 
CP MACCP、 CP OEMCP、 CP SYMBOL、CP THREAD ACP、 CP UTF7 和 CP UTFS8 


Codepage 
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在 本 项 目 中 PHP 与 Access 数据 库 的 连接 是 通过 conn.php 文件 完成 的 ， 其 中 Access 数据 库 文件 存 


储 在 根 目 录 下 的 data/db_mail shortnote mdb 文件 中 。 实 现 PHP 与 Access 数据 库 连 接 的 程序 代码 如 下 : 


<?php 

$conn = new com("adodb.connection"): /应 用 COM 类 

/获取 数据 库 文件 

$connstr="driver= {Imicrosoft access driver (*.mdb)}: dbq=". realpath("data/db mail shortnote mdb"): 
Sconn->open($connstr): /| 执行 操 作 

?> 


在 完成 数据 库 的 连接 后 ， 并 不 能 直接 对 数据 库 进 行 查询 、 更 新 和 删除 等 操作 ， 必 须 先 建立 记录 集 。 


代码 如 下 : 
include("conn.php"); // 包 含 数据 库 连 接 文 件 
Srs=new com("adodb.recordset"): // 建 立 记录 集 


S$rs->open("select * from score where sno=".$_POST[sno].”",$conn,3,1); // 执 行 查询 


上 述 代码 涉及 到 记录 集 游标 的 使 用 ， 以 $rs->open($sql,$conn,A,B) 为 例 ， 参 数 A、B 取 不 同 的 值 使 


游标 具有 不 同 的 属性 ， 如 表 9.22 和 表 9.23 所 示 。 


表 9.22 参数 A 的 用 法 
人 A 的 取 值 用 法 
0 读 ， 当 前 记录 集 指针 只 能 下 移 


和 ， 
2 可 读 写 ， 当 前 记录 集 指针 可 自由 移动 
3 可 读 写 ， 当 前 记录 集 指针 可 自由 移动 并 可 看 到 新 增 记录 


表 9.23 参数 B 的 用 法 


也 


B 的 取 值 用 法 
1 默认 锁定 类 型 ， 记 录 集 只 读 ， 不 能 修改 记录 
翡 观 锁定 ， 当 修改 记录 时 ， 数 据 提供 者 将 尝试 锁定 记录 以 确保 成 功 地 编辑 记录 ， 只 要 编辑 一 开始 ， 
则 立即 锁定 记录 
3 乐观 锁定 ， 直 到 用 update 方 法 提交 更 新 的 记录 时 才 锁 定 记录 


4 批量 乐观 锁定 ， 允 许 修改 多 个 记录 ， 只 有 调用 update batch 方 法 时 才 锁定 记录 
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.12.2 ”邮件 群发 技术 


在 本 项 目 中 ， 不 但 可 以 实现 邮件 的 单独 发 送 ， 而 且 实 现 了 邮件 群发 的 功能 。 下 面 对 邮 件 群 发 功能 


的 原理 和 实现 方法 进行 详细 讲解 。 


邮件 群发 的 实现 原理 : 首先 将 邮箱 地 址 添加 到 文本 框 中 ， 并 且 以 “*” 来 分 隔 邮 箱 地 址 ， 然 后 将 表 


单 中 的 数据 提交 到 处 理 页 ， 最 后 在 处 理 页 中 获取 文本 框 中 提交 的 邮箱 地 址 数据 ， 获 取 到 的 数据 是 一 个 
以 * 进 行 分 隔 的 字符 串 ， 这 时 应 用 explode0 函 数 对 获取 到 的 字符 串 进行 分 隔 ， 以 “*” 为 分 隔 符 ， 将 返 


回 


一 个 数组 ; 应 用 listO 函 数 读 取 数 组 中 的 元 素 ， 最 后 应 用 while 循环 语句 对 数组 中 的 元 素 进 行 输出 ， 
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并 执行 邮件 的 发 送 。 
邮件 群发 功能 实现 的 完整 代码 可 以 参考 本 章 中 的 “发 送 邮件 的 实现 过 程 ”一 节 相 关内 容 ， 这 里 
给 出 实现 邮件 群发 的 关键 代码 。 


<?php 


| 


if($_POST[submit]!=""){ // 判 断 提 交 的 值 是 否 为 空 

放 +#9#y9yy 和 这 里 省 略 了 部 分 代码 ， 主 要 介绍 如 何 实现 邮件 群发 的 实现 方法 **###s*yssyyss / 
$data=trim($_POST[touser]): // 去 除 收 件 人 字符 串 中 的 空格 
$datas=explode("*",$data); // 对 表单 中 提交 的 邮箱 地 址 信息 以 * 进 行 分 隔 ， 并 返回 一 个 数组 


执行 list 函 数 读 取 数组 中 邮箱 地 址 数据 ， 应 用 while 语 句 输出 邮箱 地 址 ， 并 在 while 语 句 内 部 执行 邮件 发 送 的 操作 */ 
while(list($name,$value)=each($datas)) { 

S$sendes=imap_mail($value,$subject,$msgbody,$Smsgheader); ” // 执 行 imap_mail0 函 数 实现 邮件 的 发 送 
include_once("conn/conn.php"): // 连 接 数 据 库 , 将 邮件 的 发 送信 息 存 储 到 数据 库 中 


$sql="insert into 


,".$mail_ date.™)"; 
Srs=new com("adodb.recordset"): 
Srs->open($sql,$conn,3,1); 


/村 率 素 事 束 事 惠 事 率 素 束 束 率 束 束 束 束 束 吕 寂 束 下 字 束 束 束 束 事 宙 宙 训 下 市 吕 训 训 束 事 束 束 下 可 事 可 下 可 可 可 束 训 训 束 中 束 束 呈 可 事 吕 事 束 训 事 束 训 中 事 李 率 事 可 束 束 事 中 束 事 可 下 束 束 ) 


?> 


9.13 ”通过 fsockopenO) 函 数 发 送 短 信 技 术 专 题 


实现 短信 发 送 的 方式 很 多 ， 不 仅 可 以 使 用 Web Service 技术 来 实现 ， 还 可 以 通过 短信 猫 或 者 
fsockopen0 函 数 来 完成 。 通 过 fsockopen0 函 数 发 送 短 信 ， 就 是 采用 socket 编程 的 方式 发 送 短信 。 用 户 
需要 做 的 是 到 移动 、 联 通 等 单位 申请 短信 网 关 和 短信 端口 ， 与 程序 实现 连接 来 发 送 短信 。 

下 面 介绍 fsockopen0) 函 数 。 该 函数 的 语法 格式 如 下 : 

int fsockopen( string hostname, int port [, int errno [, string errstr [. float timeout]]]) 


fsockopen0 函 数 的 参数 说 明 如 表 9.24 所 示 。 
表 9.24 fsockopen() 函 数 的 参数 说 明 


参 。 数 说 明 
hostname 服务 器 地 址 

port 服务 器 端口 号 

emmo | 可 省 参数 ， 连 接 服务 器 时 ， 如 果 出 错 ， 则 保存 错误 号 
emrstr | 可 省 参数 ， 连 接 服务 器 时 ， 如 果 出 错 ， 则 保存 错误 信息 


timeout 可 省 参数 ， 连 接 服务 器 的 最 大 超时 时 间 


下 面 详细 讲解 通过 socket 编程 实现 短信 发 送 技术 。 操 作 步骤 如 下 : 
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(1) 创建 一 个 表单 页 socketphp， 用 于 提交 网 关 地 址 、 端 口号 、 用 户 名 、 密 码 、 发 送 受 机 号 码 、 


接收 受 机 号 码 和 短信 内 容 ，socket.php 页 中 表单 的 重要 属性 如 表 9.25 所 示 。 
表 9.25 发送 短信 中 使 用 的 重要 表单 元 素 


名 称 元 素 类 型 重要 属性 含义 
forml form method="post" action="socket.php" onsubmit="retum chkinput(this)" 表单 
ip text Class="inputcss" size="25" 网 关 地 址 
_port text size="5" class="inputcss”" 端口 号 
username text class="inputcss" size="25" 用 户 名 
userpwd password Class="inputcss" size="25" 密码 
telnumber text Class="inputcss" size="25" 发 送 手 机 号 码 
receivenumber text Class="inputcss" size="25" 接收 手机 号 码 
content textarea Cols="40" rows="5" class="inputcss" 短信 内 容 
Submit class="buttoncss" id="submit" value=" 发 送 " 提交 表单 


(2) 在 socketphp 页 中 通过 $_ POST[ ] 全 局 数组 接收 表单 提交 的 信息 , 最 后 通过 文件 操作 函数 将 短 


信 发 送出 去 。 关 键 代 码 如 下 : 


<?php 
if($_POST[submit]!=""){ 
S$smsUID=$_ POST[username]: /短信 网 关 分 配 的 用 户 名 和 密码 
S$smsPWD=$_ POST[userpwd]: 
S$smsSocket=$ POST[id]: // 短 信和 网 关 的 他 
$smsPost=$_POST[port]: // 短 信和 网 关 的 端口 
$fp=(@fsockopen($smsSocket,$smsPost,&$errmo, &S$errstr, $smsTimeout); 
这 !$fp){ 
echo "<script>alert(' 与 短信 网 关连 接 失 败 !"):</script>"; 
yelse{ // 否 则 登录 到 短信 中 心服 务 器 
fputs($fp,"login\n"): /将 loginm 写 入 到 文件 $p 中 


fputs($fp.$smsUID."\n"): 
fputs($fp,$smsPWD."\n"): 


fputs(Sfp."\n"): 
$MessageContent=trim($_POST[content]); 。“ /获取 短信 和 内容 
$MobileNo=trim($_POST[telnumber]): // 获 取 电 话 号 码 
$ReceiveNo=trim($_ POST[receivenumber]): /获取 接收 电话 号 码 
SServiceType="MFFW": // 计 费 代码 TP0.5 按 条 收费 

// 发 送 优先 级 

// 代 收费 标志 

1/ 点播 号 

// 短 信和 失效 时 间 

// 定 时 发 送 时 间 

3 // 状 态 报告 

Sstatus="255"; // 都 要 返回 状态 报告 
SMessageType="TEXT": // 短 信 类 型 ， 文 本 信息 
SFreeTerminalNo=$MobileNo: // 记 费 手机 号 码 ， 本 实例 采用 收 短信 方 收费 
STargetTerminalNo=$ReceiveNo:; // 接 收 方 手机 号 码 
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S$SourceTerminalNo=$MobileNo:; // 发 送 方 手机 号 码 
$Messageld="123"; 

/二 本 二 二 本 本 本 二 直 本 本 中 汪 * 本 本 有 人 7 用 文件- 系统 函数 对 信息 进行 编译 并 发送 ## 地 # 本 本 本 本 本 本 下 林 本 本 
Print(fgets($ 印 ,4096)): 

Print(fgets($ 印 .4096)): 

fputs(Sfp,"submit' wm: 

fputs(Sfp.SMessageId mr: 

fputs($fp,SFreeTerminalNo."\n"); 

fputs($fp,$SourceTerminalNo."\n"); 

fputs($fp,$TargetTerminalNo."\n"): 

fputs($fp,SServiceType."\n"): 

fputs($fp,SMoFlag."\n"): 

fputs($fp,SReportFlag."\n"): 

fputs(Sfp,SExpireTime."\n"): 

fputs($ 印 .SScheduleTime mn): 

fputs($fp,SMessageType."\n"); 
$MessageContent=str_replace("\","",str_replace("\n",."",$MessageContent)); // 不 能 有 回 车 
fputs($fp,S$MessageContent."\n"); 

fputs($fp,"\n"); 

print("<div align=center> 发 送 成 功 !".SMobileNo." : ".SMessageContent."</div>"); 
felose(Sfp): 


192. 168. 1. ##* 器 : 


3334## 
新 年 快乐 


图 9.48 socket 编程 发 送 短信 的 运行 结果 


9.14 本 章 总 结 


本 章 主要 讲解 的 是 企业 快 信 系统 的 开发 过 程 ， 从 最 初 的 系统 分 析 ， 到 系统 的 设计 ， 再 到 数据 库 的 
设计 ， 最 后 到 具体 模块 的 创建 。 其 中 在 具体 模块 创建 的 过 程 中 针对 系统 信息 管理 模块 、 发 送 短信 模块 、 


连接 邮件 接 


技巧 。 例 如 ， 


模块 、 接 收 邮件 模块 和 发 送 邮件 模块 进行 重点 讲解 ， 并 且 在 其 中 应 用 到 很 多 新 的 技术 和 


通过 Web Service 实现 发 送 短信 技术 、 通 过 fsockopen0 函 数 实 现 发 送 短 信 技 术 、 通 过 imap 


电子 邮件 系统 函数 实现 发 送 和 接收 邮件 技术 , 以 及 通过 ADO 实现 PHP 与 Access 数据 库 的 连接 技术 等 。 


和 


TOs 
第 章 
online 影视 365 网 


(Apachet+PHP+ADODBt+phpMyAdmin+MySQL 5.0 实现 ) 
( 铭 s 视频 讲解 : 1 小 时 31 分 钟 ) 


在 全 球 知 识 经 济 和 信息 化 高 速 发 展 的 今天 ,网 络 化 是 企业 发 展 的 趋势 ，21 世纪 
的 人 更 习惯 在 网 站 上 听 音 乐 . 看 电影 ,所 以 企业 要 在 同 领域 中 得 到 突飞猛进 的 发 展 ， 
就 必须 借助 网 络 。 

当今 社会 进入 了 一 个 信息 快速 发 展 的 社会 ,在 网 络 世 界 浏览 信息 的 同时 也 可 以 
听 听 和 歌曲， 这 样 既 愉悦 了 身心 ， 又 得 到 了 最 新 的 市 场 动态 ， 由 此 网 络 上 也 出 现 了 很 
多 的 影视 网 站 ， 都 很 受到 欢迎 。 未 来 视听 生活 的 新 空间 ， 也 必然 在 宽带 互联 网 上 开 
启 。VOD 的 概念 已 经 被 越 来 越 多 的 人 所 接受 ， 逐渐 成 为 网 络 发 展 的 必然 趋势 之 一 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


online 影视 365 类 网 站 的 整体 设计 思路 及 实现 方法 
应 用 <embed> 和 <object> 播 放 影 音 文件 

关闭 子 窗口 时 自动 刷新 父 窗口 

制作 无 边框 的 窗口 

ADODB 的 介绍 及 使 用 

更 改 数据 库 时 的 注意 事项 


豆 吾 于 于 至 至 
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10.1 开发 背景 


X XX 影视 有 限 公司 是 一 家 以 影视 为 主 的 公司 ， 为 了 扩大 企业 规模 ， 增 强 企 业 的 竞争 力 ， 公 司 决 
定向 多 元 化 发 展 。 计 算 机 技术 、 网 络 通信 技术 、 多 媒体 技术 及 数据 存储 技术 的 飞速 发 展 ， 对 人 类 的 生 
产 和 生活 方式 产生 了 很 大 影响 。 网 络 传播 以 其 特有 的 快速 、 高 效 、 便 捷 的 传输 方式 直接 被 人 们 所 接受 。 
随 着 多 媒体 数据 的 存储 、 传 输 和 应 用 技术 的 不 断 成 熟 ， 以 及 宽带 网 络 的 不 断 发 展 ， 我 们 有 理由 相信 和 宽 
带 点 播 一 定 会 成 为 网 络 内 容 创 新 的 重头 戏 。 


10.2 需求 分 析 


根据 客户 要 求 及 成 本 核算 ， 本 系统 所 要 实现 的 功能 如 下 : 

网 站 实行 会 员 等 级 管理 ， 分 为 “普通 会 员 ” 和 “高 级 会 员 ” 两 种 。 

网 站 提供 下 载 功能 和 在 线 视听 功能 ， 不 同 的 会 员 等 级 实现 的 功能 操作 也 不 同 。 

网 站 分 为 前 台 和 后 台 两 部 分 ， 没 有 权限 ， 用 户 无 法 进入 后 台 进行 管理 。 

独特 的 点 歌 模块 ， 为 网 站 的 会 员 提 供 点 歌 平台 。 

前 期 数据 量 比较 少 ， 可 以 使 用 小 型 数据 库 以 节约 成 本 ， 如 果 后 期 数据 量 增 大 ， 则 可 以 更 换 大 
型 数据 库 。 


办 办 办 多 凶 


10.3 系统 分 析 


10.3.1 系统 目标 


online 影视 365 网 主要 实现 以 下 目标 : 

自 定义 的 鼠标 样式 增强 网 站 界面 的 和 谐 性 。 

采用 无 边框 窗口 技术 ， 保 障 系统 的 安全 性 。 

信息 搜索 灵活 、 方 便 。 

采用 多 媒体 引用 技术 播放 影音 文件 。 

网 站 采用 文件 上 传 和 下 载 技术 实现 图 像 、 音 频 、 视 频 文件 的 客户 端 上 传 和 下 载 。 


10.3.2 ”系统 功能 结构 


办 办 办 凶 轨 


根据 online 影视 365 网 的 特点 ， 可 以 将 其 分 为 前 台 和 后 台 两 个 部 分 设计 。 前 台 主 要 实现 在 线 视听 、 
影视 音乐 上 传 、 影 视 音乐 下 载 、 在 线 聊 天 和 在 线 点 歌 等 功能 ， 后 台 主 要 用 于 管理 员 对 影视 音乐 目录 、 
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数据 信息 和 上 传 日 志 进 行 管理 。 
online 影视 365 网 的 前 台 功 能 结构 图 如 图 10.1 所 示 。 


online 影 视 365 网 前 台 


I [ I 
在 线 视听 ] [上 上传 专区 ] [点 歌 专区 | [详细 信息 | [信息 展示 ] [信息 查询 ] [下 载 排行 | [ 会 员 专区 ] [站 长 登录 ] 
I 

| | | 

视 视 | [ 音 | 会 [会 后 
在 | | 在 | | 频 一 | | 高 | | 频 | | 颇 | 员 | 员 | 会 | 会 | | 全 
线 || 线 | | 数 般 || 级 | | 下 || 下 | 密 | 资 | 员 | 员 | | 登 
视 | | 观 | “| 所 查 | | 查 | “| 载 | | 载 | 码 | 料 | 注 | 登 | | 录 
听 || 看 | | 上 询 | | 询 | | 排 | | 排 | 找 | 修 | 由 | 录 | | 管 

伟 行 | | 行 | 加 | 改 理 


图 10.1 online 影视 365 网 前 台 功能 结构 图 
online 影视 365 网 的 后 台 功 能 结构 图 如 图 10.2 所 示 。 


online 影 视 365 网 后 台 


上 赔 于 眶 炎 湾 党 


El 
传 | | 传 
日 日 
志 | | 志 
查 | | 浏 
询 | | 览 


上 崩 趴 郊 耿 泪 藻 
并 碎 济 卫 江东 


图 10.2 ”online 影视 365 网 后 台 功能 结构 图 


10.3.3 功能 预览 


onlie 影视 365 网 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 
个 典型 功能 的 页 面 ， 其 他 页 面相 关内 容 可 参见 光盘 中 的 源 程序 。 

影视 专区 页 面 如 图 10.3 所 示 , 该 页 面 展 示 所 有 的 视频 信息 及 其 分 类 。 音乐 专区 页 面 如 图 10.4 所 示 ， 
该 页 面 展 示 所 有 的 音频 信息 及 其 分 类 。 

最 新 影视 页 面 如 图 10.5 所 示 ， 该 页 面 用 于 显示 所 有 的 最 新 上 传 的 影视 文件 。 注 册页 面 如 图 10.6 所 
示 ， 该 页 面 显示 了 游客 注册 时 需要 填写 的 昵称 、 密 码 和 个 人 资料 等 相关 内 容 。 

查询 页 面 如 图 10.7 所 示 , 该 页 显示 简单 查询 所 查 到 的 音频 、 视频 文件 列表 。 高 级 查询 页 面 如 图 10.8 
所 示 。 该 页 面 用 于 模糊 查询 音 、 视 频 文件 ， 包 含 了 多 个 字段 ， 用 户 可 任意 选择 。 
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图 10.3 ”影视 专区 (光盘 \TM\10\online\listphp?action=audio) 图 10.4 音乐 专区 (光盘 \TM\1Ovonline\listphp?action=video) 


图 10.6 注册 页 面 〈 光 盘 \TMNMIO\online\ register.php) 


| 


图 10.7 简单 查询 (光盘 \TM\10\online\show.php) 图 10.8 ”高 级 查询 (光盘 \TM\10\online\high php) 
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10.3.4 系统 流程 图 


online 影视 365 网 的 系统 流程 图 如 图 10.9 所 示 。 


上 传记 录 查 看 


图 10.9 online 影视 365 网 的 系统 流程 图 


10.3.5 ”开发 环境 


在 开发 online 影视 365 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
， 服 务 器 端 
操作 系统 : Windows Server 2003。 
服务 器 : Apache 2.0。 
PHP 软件 : PHP 5.0。 
数据 库 : MySQL 5.0。 
MySQL 图 形 化 管理 软件 : PhpMyAdmin-2.5.5。 
开发 工具 : Dreamweaver 8。 
浏览 器 : IE 6.0。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 
.客户 端 
浏览 器 : 推荐 使 用 正 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效 果 1024X768 像素 。 


QREAARRARAAREAR= 


加 因 个 


10.3.6 文件 夹 组 织 结构 


online 影视 365 网 包括 前 台 和 后 台 两 部 分 ， 所 以 文件 夹 结构 也 主要 由 两 部 分 组 成 。online 影视 365 
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网 的 文件 夹 组 织 结构 图 如 图 10.10 所 示 。 
日 岛 oline 一 “影视 365 系 统 根 目录 
日 加 sanin 一 -后台 管理 系统 目录 
EB 后 台数 据 库 链 接 文件 目录 
日 仿 一 后 台 css 样 式 文件 目录 
田 筷 s 后 台 背 景 及 图 标 文件 目录 
日 入 后 各自 定义 函数 及 公共 文件 目录 
日 入 后 台 javaseript 脚 本 文件 目录 
田 国 at- 一 ado 了 hb 库 文件 目录 
四 -入 cm 一 前 台数 据 库 链接 文件 目录 
田 国 css- 前 台 css 样 式 文件 目录 
田 局 images ~- 前 台 背 县 及 图 标 目录 
田 - 国 ine 前 台 自 定义 函数 及 公共 文件 目录 
四 国 js 前 台 ijsvasceript 脚 本 文件 目录 
田 铝 wfiles ~- 上 传 文件 管理 目录 
图 10.10 ”online 影视 365 网 的 文件 夹 组 织 结构 图 


10.4 ”数据 库 设计 


10.4.1 ”数据 库 分 析 


本 系统 使 用 的 是 MySQL 数据 库 ， 但 使 用 ado 的 连接 方式 ， 这 是 为 了 便于 以 后 当 数据 负担 加 重 时 ， 
只 需 更 改 其 他 数据 库 〈 如 oracle) ， 而 无 须 重 新 编写 源 程序 。 


10.4.2 数据库 概 念 设计 


通过 需求 分 析 和 功能 上 的 设计 ， 本 系统 规划 出 管理 员 信息 实体 、 会 员 信息 实体 、 视 频 信 息 实 体 、 
音频 信息 实体 、 视 频 目 录 实 体 和 音频 目录 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 

会 员 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 下 载 或 在 线 视 听 ， 则 必须 注册 为 会 员 才 可 以 。 
会 员 信息 实体 E-R 图 如 图 10.11 所 示 。 


图 10.11 会 员 信息 实体 E-R 图 
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视频 信息 实体 包括 视频 名 称 、 视 频 图 片 、 视 频 文件 、 主 要 演员 、 导 演 、 发 行商 等 多 项 资料 。 视 频 
信息 实体 E-R 图 如 图 10.12 所 示 。 
音频 信息 实体 包括 音频 目录 分 类 的 相关 信息 ， 如 目录 名 称 ， 目 录 级 别 、 父 目录 名 称 等 。 音 频 信 息 


实体 的 E-R 图 如 图 10.13 所 示 。 
目录 级 别 


图 10.12 ”视频 信息 实体 E-R 10.13 ”音频 信息 实体 E-R 


10.4.3 ”数据库 物理 结构 设计 


创建 时 间 


online 影视 365 网 的 数据 库 中共 包 含 8 个 数据 表 ， 如 图 10.14 所 示 。 


加 服务 器 : localhost ， 轧 数据 库 : db_online 

表 整理 说 明 
tb_account ”MyISAM ”gb2312_chinese_cl ”会员 信 息 列表 
tb_audio MyISAM ”gb2312_chinese_ci ”视频 信息 列表 
tb_audiolist 。 MylSAM ”gb2312_chinese_ci ”视频 目录 列表 
th_grade MyISAM ”gb2312_chinese_ci ”等 级 限定 列表 
tb_manager MylSAM ”gb2312_chinese_cl ”管理 员 列 表 
也 _register 。 MylSAM gb2312_chinese_ci 。 点 歌 信息 列表 
th_video MyISAM ”gb2312_chinese_ci ”音频 信息 列表 
tb_videolist ”MylSAM ”gb2312_chinese_ci ”音频 目录 列表 


10.14 db_online 库 中 的 数据 库 列表 
下 面 来 看 一 下 各 个 表 的 结构 设计 。 
1. tb_account (会 员 信息 列表 ) 
会 员 信息 列表 主要 存储 用 户 的 个 人 信息 ，tb_account 表 的 结构 如 图 10.15 所 示 。 
2. tb_audio 〈 视 频 信息 列表 ) 
视频 信息 列表 主要 上 传 视频 的 资料 , 如 视频 名 称 、 视 频 图 片 等 。 tb_audio 表 的 结构 如 图 10.16 所 示 。 
3. tb_audiolist (视频 目录 列表 ) 
视频 目录 列表 主要 是 上 传 视频 所 属 的 类 型 (二 级 目录 〉 及 类 别 ( 一 级 目录 ) 。tb_audiolist 表 的 结 
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构 如 图 10.17 所 示 。 


胃 服务 器 : localhost > 晶 数据 库 : db_online 图 表 :tb_account 


字 且 类 型 整理 尾 性 Null 默认 说 明 
id int(11) 否 自动 编号 
name varchar(30) gb2312_chinese_ci 否 会 员 昵称 
password varchar(30) gb2312_chinese_ci 否 会 员 密 码 
question 。 varchar(50) gb2312_chinese_ci 理 密码 提示 问题 
answer varchar(50) gb2312_chinese_ci 否 密码 提示 答案 
realName varchar(30) gb2312_chinese_ci 否 会 员 真实 姓名 
numbers varchar(20) gb2312_chinese_ci 否 身份 证 号 
sex varchar(10) gb2312_chinese_ci 否 会 员 性 别 
age int(4) 否 会 员 年 龄 
job varchar(50) gb2312_chinese_ci 否 所 做 工作 
email varchar(100) gb2312_chinese_ci 要 了 -mail 
address 。 varchar(100) gb2312_chinese_cl 否 联系 地 址 
phone varchar(20) 。 gb2312_chinese_ci 理 联系 电话 
qq varchar(20) ”gb2312_chinese_ci 否 联系 99 
http varchar(100) gb2312_chinese_ci 否 个 人 主页 
counts int(4) 否 0 上 传 次 数 
grade varchar(10) 。 gb2312_chinese_ei 否 。 普通 会 员 会 员 级 别 
whether varchar(10) gb2312_chinese_ci > 是 否 激活 


图 10.15 会 员 信息 列表 结构 


胃 服务 器 : localhost 加 数据 库 : db_online ， 国 表 :tb_audio 


字段 类 型 整理 尾 性 Null 默认 说 明 
这 int(4) 否 自动 编号 
name varchar(100) gb2312_chinese_ci 否 视频 名 称 
picture varchar(200) 。 gb2312_chinese_ci 否 封面 图 片 
sizes varchar(50) gb2312_chinese_ci 理 视频 大 小 
grade varchar(10) gb2312_chinese_ci 理 视频 级 别 
publisher varchar(100) gb2312_chinese_cl 否 发 行 公司 
actor varchar(200) 。 gb2312_chinese_el 否 主要 演员 
director varchar(100) gb2312_chinese_ci 否 导演 
marker varchar(100) 。 gb2312_chinese_ci 否 制 片 
languages varchar(20) gb2312_chinese_ci 否 语种 
type varchar(50) gb2312_chinese_ci 否 类 型 
style varchar(50) 。 gb2312_chinese_ci 否 类 别 
froms varchar(100) gb2312_chinese_ci 否 视频 所 属地 区 
publishTime date 和 否 发 行 时 间 
remark varchar(1000) gb2312_chinese_ci 否 视频 介绍 
property varchar(20) 。 gb2312_chinese_ci 否 视频 属性 
address varchar(200) gb2312_chinese_ci 否 存储 地 址 
username varchar(50) gb2312_chinese_ci 否 发 布 人 姓名 
issueDate datetime 否 发 布 时 间 
downTime int(4) 否 0 下 载 次 数 
lastTime cdatetime 否 景 后 下 载 时 间 
whether varchar(20) gb2312_chinese_ci | 是 否 为 新 品 


图 10.16 视频 信息 列表 结构 
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盟 服务 器 : localhost ， 四 数据 库 : db_online 国 表 :tb_audiolist 
字 及 类 型 整理 屋 性 ”Null 葡 认 说 明 
于 int(4) 否 自动 编号 
grade varchar(20) gb2312_chinese_ci 否 目录 级 别 
name varchar(50) gb2312_chinese_ci 否 目录 名 称 
father varchar(50) gb2312_chinese_ci 否 妈 馈 目录 
userName varchar(30) gb2312_chinese_ci 否 创建 人 昵称 
issueDate = datetime 否 自 障 时 间 


图 10.17 视频 目录 列表 结构 
4. tb_manager (管理 员 列表 ) 


管理 员 列 表 主 要 用 于 后 台 管 理 员 的 资料 信息 管理 ， 包 括 管理 员 名 称 、 管 理 员 权限 等 。tb_manager 
表 的 结构 如 图 10.18 所 示 。 


胃 服务 器 : localhost 加 数据 库 : db_online ， 国 表 :tb_manager 
宇和 & 类 型 整理 尾 性 Null 菊 认 说明 
到 int(4) 否 自动 编号 
name varchar(50) gb2312_chinese_ci 理 管理 员 昵称 
password varchar(50) gb2312_chinese_ci 否 管理 员 密码 
type varchar(50) gb2312_chinese_ci 否 管理 员 关 型 
realName 。 varchar(50) gb2312_chinese_ci 否 真实 姓名 
issueDate datetime 理 创建 时 间 
whether 。 varchar(50) gb2312_chinese_ci 否 是 否 激活 


图 10.18 管理 员 列 表 结 构 
5. tb_register (点 歌 信息 列表 ) 


点 歌 信息 列表 主要 存储 会 员 用 户 之 间 的 点 歌 祝 福 的 信息 ， 主 要 包括 发 送 人 、 接 收 人 人 、 歌 曲 地 址 等 。 
tb_register 表 的 结构 如 图 10.19 所 示 。 


加 服务 器 : localhost ， 轧 数据 库 : db_online ， 国 表 :tb_register 
字 及 类 型 整理 尾 性 Null 默认 说 明 
地 int(4) 否 自动 编号 
name nt) 否 歌曲 地 址 
fromName varchar(50) gb2312_chinese_ci 否 发 送 人 姓名 
toName 。 varchar(50) gb2312_chinese_ci 理 接收 人 姓名 
remark varchar(50) gb2312_chinese_ci 二 备注 
issueDate datetime 否 点 歌 时 间 


图 10.19 点 歌 信 息 列表 结构 
6. tb_video (音频 信息 列表 )》) 
音频 信息 列表 主要 用 于 音频 文件 的 信息 管理 , 如 歌手 、 歌词 等 。 tb_video 表 的 结构 如 图 10.20 所 示 。 
7. tb_videolist (音频 目录 列表 》) 


音频 目录 列表 主要 用 于 音频 文件 的 分 类 管理 ， 包 括 目录 名 称 、 目 录 级 别 等 。tb_videolist 表 的 结构 
如 图 10.21 所 示 。 
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困 服务 器 :localhost ”总 要 据 库 : db_online ， 国 表 :tb_video 
字段 类 型 整理 性 性 Nul 默认 说 明 

过 int(4) 否 自动 编号 
name varchar(100) ”gb2312_chinese_ci 否 音频 名 称 
picture varchar(200) 。 gb2312_chinese_ci 否 封面 图 片 
actor varchar(100) gb2312_chinese_ci 晋 主唱 
ci varchar(50) gb2312_chinese_ci 否 歌词 作者 
gu Yarchar(50) gb2312_chinese_ci 否 歌曲 作者 
actortype varchar(50) gb2312_chinese_ci 否 歌手 类 型 
style Yarchar(50) gb2312_chinese_ci 否 音频 类 别 
publisher varchar(100) gb2312_chinese_ci 否 发 行商 
froms varchar(100) gb2312_chinese_ci 否 发 行 地 区 
type Yarchar(50) 。 gb2312_chinese_ci 否 音频 类 型 
sizes varchar(50) 。 gb2312_chinese_ci 否 音频 大 小 
languages 。 varchar(20) gb2312_chinese_ci 否 音频 语言 
publishTime date 否 发 行 时 间 
remark Yarchar(1000) gb2312_chinese_ci 否 备注 
property varchar(20) 。 gb2312_chinese_ci 理 音频 属性 
address varchar(200) 。 9b2312_chinese_ci 否 音频 地 址 
userName varchar(50) gb2312_chinese_ci 理 发 布 人 姓名 
issueDate 。 datetime 否 发 布 时 间 
downTime int(4) 否 下 载 次 数 
lastTime datetime 否 最 后 下 载 时 间 
whether varchar(50) gb2312_chinese_ci 否 1 是 否 为 新 品 


图 10.20 音频 信息 列表 结构 


胃 服务 器 : localhost 加 数据 库 : db_online ， 国 表 :tb_videolist 
字 有 & 类 型 整理 尾 性 Null 默认 说 明 
过 int(4) 否 自动 编号 
grade varchar(50) gb2312_chinese_ci 和 理 目录 等 级 
name varchar(50) gb2312_chinese_ci 否 目录 名 称 
father varchar(50) gb2312_chinese_cl 否 你 冉 目 录 
userName varchar(50) gb2312_chinese_ci 天 创建 人 姓名 
issueDate datetime 否 创建 时 间 


| 


图 10.21 音频 目录 列表 结构 
10.5 前 台 首 页 设计 


10.5.1 前 台 首页 概述 


online 影视 365 网 的 前 台 首 页 的 功能 模块 主要 包括 以 下 3 部 分 : 

网 站 首部 导航 栏 : 主要 有 影视 专区 、 音 乐 专区 、 上 传 专区 和 点 歌 专区 。 
回 ”网 站 左 侧 导航 栏 : 包括 会 员 登 录 模 块 、 搜 索 模块 和 影视 排行 模块 。 

回 ”网 站 主 显示 区 : 包括 最 新 上 传 模块 、 视 频 模块 和 音频 模块 。 
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下 面 看 一 下 本 案例 中 提供 的 前 台 首页 , 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\10\online\index.php， 
如 图 10.22 所 示 。 


eonines 向 365 


【aas] 矿 采 水 ** 


图 10.22 ”online 影视 365 前 台 首页 
10.5.2 ”前 台 首 页 技术 分 析 


在 本 章 的 数据 库 分 析 中 已 经 提 到 ， 本 系统 所 使 用 的 连接 数据 库 的 方式 为 ADODB 连接 。 这 里 ， 先 
来 初步 了 解 一 下 ADODB 如 何 连 接 MySQL。 
下 面 以 本 系统 使 用 的 数据 库 和 数据 表 为 例 来 介绍 使 用 ADODB 的 基本 操作 步骤 。 
(1) 载 入 (include) adodb.in.php 文件 。 


include "../ado/adodb.inc.php": 
(2) 建立 数据 库 类 型 连接 。 
$com = &ADONewConnection("mysq"): or S$conn = &NewADOConnection(“"mysql"): 


(3) 连接 要 使 用 的 数据 库 。 
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S$conn->PConnect("db_online"): or Sconn->Connect("db online"): 
(4) 执行 SQL 语句 。 

Srecordset = $conn->Execute("select * from tb_audio"): 
(5) 判断 结果 ， 并 对 结果 处 理 。 


if($recordset) // 判 断 execute0 函 数 执行 中 是 否 有 错误 
while(!$recordset->EOF){ // 如 果 没 有 错误 ， 则 配合 wihle 语 句 循环 输出 结果 
echo Srecordset->fields[0]."<br>"; 
Srecordset->movenext(); // 指 针 下 移 
1 
} 


(6) 关闭 连接 。 
Srecordset->close():; 
$conn->close(): 


如 果 是 对 微软 的 ado 有 所 了 解 的 用 户 ， 会 对 上 述 步骤 有 一 种 亲切 感 。 因 为 无 论 从 字段 名 称 ， 还 是 


从 功能 上 ， 两 者 都 非常 相似 ， 所 以 上 手 会 很 轻松 。 没 有 接触 过 aod 的 用 户 也 不 用 心急 ， 用 ADODB 操 
作 数 据 库 主要 就 是 这 6 个 步骤 ， 只 要 记 住 这 个 顺序 ， 练 习 几 遍 后 ， 也 会 运用 自如 。 


在 本 章 的 最 后 对 ADODB 类 库 进行 了 较 详细 的 讲解 ， 参 见 10.14 节 。 


10.5.3 前台 首页 的 实现 过 程 


国 前台 首页 使 用 的 数据 表 : tb_account、tb_audio、tb_video 
在 首页 中 ， 使 用 到 了 3 个 include 语句 ， 将 导航 栏 、 登 录 框 、 搜 索 框 、 浏 览 区 等 主要 的 模块 加 载 进 


限于 篇 幅 ， 这 些 页 面 的 代码 不 做 讲解 了 ， 请 读者 自行 到 光盘 中 查找 。 前 台 首 页 的 实现 代码 如 下 : 
例 程 01 代码 位 置 : 光盘 VTM\10\online\index.php 
<?php 
Session start(): /开启 session 支 持 
?> 
<script src="js/chk.js" language="javascript"></script> /引入 js 脚本 文件 
<link rel="stylesheet" href="css/style.css" /> // 引 入 css 样 式 文件 
<?php 
include "top.php": // 首 部 导航 及 LOGO 
?> 


<table border="0" cellpadding="0" cellspacing="0"> 
<tr><td width="265" align="center" valign="top"> 


<2php 

include "left.php": // 登 录 、 搜 索 框 
?> 

</td><td width="605" align="center' valign="top"> 
<?php 

include "main php": // 主 浏览 区 
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?> 
</td></tr></table> 


10.6 详细 信息 查看 模块 


10.6.1 信息 查看 模块 概述 


详细 信息 查看 模块 主要 包括 视频 详细 信息 查看 
和 音频 详细 信息 查看 两 个 部 分 。 查看 信息 功能 没有 权 
限 限制 ， 无 论 是 游客 还 是 会 员 都 可 以 查看 。 但 查看 页 Ca ) 


有 
了 


面 中 的 下 载 和 “在 线 视听 ”按钮 则 是 只 有 会 员 才 可 以 ~、、 
看 到 的 操作 ， 游 客 只 能 看 到 “返回 ”按钮 ， 用 来 关闭 SS 
查看 的 页 面 。 下 面 给 出 信息 查看 框架 图 ， 如 图 1023 


在 这 个 模块 中 ， 所 要 解决 的 技术 难点 就 是 : 
需要 根据 不 同 的 浏览 用 户 ， 显 示 不 同 的 操作 权限 。 这 
里 是 使 用 站 语句 ， 配 合 session 来 实现 其 功能 的 。 首 图 10.23 信息 查看 框架 图 
先 通过 检测 $_SESSION[name] 是 否 存 在 ， 来 判断 用 户 是 否 登 录 ， 如 果 没 有 登录 ， 则 跳 过 “在 线 视听 ” 
和 “下 载 ”， 只 显示 “返回 ”按钮 ， 如 果 用 户 为 登录 会 员 ， 则 显示 “在 线 播放 ”按钮 ， 接 着 判断 
$_SESSION[grade] 是 否 为 “高 级 会 员 ”， 如 果 登 录 会 员 为 普通 会 员 ， 则 “下 载 ” 按 钮 为 灰色 失效 状态 ， 
如 果 登 录 会 员 为 “高 级 会 员 ”， 则 同时 显示 3 个 按钮 的 全 部 功能 。 该 功能 的 关键 代码 如 下 : 

例 程 02 ”代码 位 置 : 光盘 \TM\10vonline\introphp 

<?php 
人 # 如 果 用 户 是 登录 会 员 */ 
if($_SESSION[name]—"" ){ 


所 示 。 py 
10.6.2 ”信息 查看 模块 技术 分 析 人 
游客 


会 员 


?> 
<!-- 显示 “播放 ”按钮 --> 
oo <input name="Submit" type="submit" value=” 播 放 " 
onclick="javascript:Wopen=open('operation.php?action=see&id=<?php echo 
Srst->fields[16]; ?>",",'height=700,width=665,scrollbars=n0");"> 
<?php 
必 ”如 果 是 登录 会 员 */ 
}if($_SESSION[name]—""){ 


请 ”如果 是 高 级 会 员 ， 则 “下 载 ”按钮 为 激活 状态 ， 否 则 为 灰色 */ 
@ <input name="Submit” type="button”<?php 这 ($_SESSION[grade]<>" 高 级 会 员 "){ echo "disabled":}?> 
value="” 下 载 "onclick="javascript:Wopen=location='download.php?action=audio&id=<?php echo 
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Srst->fields[16]: ?>":"> 
<?php 
BB 


?> 
局 显示 “返回 ”按钮 */ 

@ <input name="Submit2" type="button" value="” 返回 "class="submit" 
onClick="javascript:top.window.closeO|;"></td> 

< 人 代码 贴 二 
@ open() 方 法 可 以 打开 一 个 无 边框 的 子 窗口 。 关 于 open 方法 的 介绍 ， 可 以 参考 10.13 节 。 
@ disabled: 译 为 无 效 的 、 不 起 作用 的 ， 这 里 是 button 按钮 的 属性 ， 即 不 可 单 击 状态 。 
目 top.window.close0: 关闭 弹出 窗口 。 


10.6.3 ”信息 查看 的 实现 过 程 


国 ”消息 管理 模块 使 用 的 数据 表 : tb_ audio、tb_video 
用 户 可 以 在 信息 展示 页 面 单 击 往 图 标 进入 详细 信息 展示 页 面 。 详 细 信息 展示 页 面 主要 是 根据 传递 
的 参数 进行 数据 库 检 索 ， 并 将 结果 集 输出 到 浏览 器 中 。 查 看 页 面 的 运行 结果 如 图 10.24 所 示 。 


图 10.24 详细 信息 页 面 的 运行 结果 


在 详细 信息 展示 页 面 (introphp) 中 ， 通 过 id 值 从 数据 表 中 提取 所 要 显示 的 信息 输出 到 页 面 ， 当 
文件 信息 输出 完毕 后 ， 通 过 session 值 来 判断 访问 者 是 游客 还 是 会 员 ， 如 果 是 游客 ， 则 只 显示 “返回 ” 
按钮 ， 如 果 是 会 员 ， 还 会 显示 “在 线 试听 ”和 “下 载 ” 按 钮 。 程 序 的 关键 代码 如 下 : 
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例 程 03 ”代码 位 置 : 光盘 \TMNIOVonlinevinto php 


<?php 
session start(): // 开 启 session 支 持 
include "conn/conn.php"; // 载 入 数据 库 连 接 文件 
?> 
<link rel="stylesheet" href="css/style.css"> // 引 入 css 样 式 文件 


<!-- 显示 数据 详细 信息 -> 


<?php ”$sql="select * from tb_audio where id=".$_GET[id]; /根据 id 生成 查询 语句 
Srst = $conn->execute($sq]): /执行 SQL 语句 ， 返 回 结果 集 
必 如果 结果 集 不 为 空 ， 则 输出 信息 “六 
这 !Srst->EOF){ 
?> 
<table width="400" border="0" align="center" cellpadding="0" cellspacing="0"> 
<t> 
<td height="15" colspan="2">&nbsp:</td> 
</t> 
<!-- 输出 视频 名 称 --> 
<tr> 
<td width="131" height="20" align="right" valign="middle"> 名 称 : </td> 
<td width="269" height="20"><?php echo Srst->fields[1]: ?></td> 
</t> 
<!-- 输出 文件 大 小 -> 
<t> 
<td height="20" align="right" valign="middle"> 大 小 : </td> 
<td height="20"><?php echo Srst->fields[3]: ?></td> 
</tr> 
<t> 
<td height="30" colspan="2" align="center" valign="middle"> 
局 ”根据 不 同 用 户 ， 显 示 不 同 权限 */ 


10.7 点 歌 模块 设计 


10.7.1 ”点 歌 模块 概述 


用 户 通过 主导 航 条 进入 点 歌 专区 。 在 点 歌 专区 ， 可 以 对 专区 内 的 音乐 进行 试听 、 浏 览 音频 信息 的 
详细 内 容 和 打开 点 歌 页 面 进行 点 歌 ， 进 行 点 歌 的 前 提 条 件 是 用 户 必 须 登 录 。 点 歌 模块 的 用 例 图 如 
10.25 所 示 。 
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在 线 试听 
= Csse ) -EL 


查 


祝福 语 


图 10.25 ”点 歌 模块 的 用 例 图 
10.7.2 ”点 歌 模块 技术 分 析 


本 系统 的 在 线 点 歌 模块 实现 了 会 员 间 发 送 祝福 与 点 歌 的 功能 , 收 到 祝福 的 会 员 可 以 实现 在 线 听 歌 。 
本 节 和 10.8 节 的 “在 线 视听 ”模块 中 ， 将 逐一 介绍 实现 在 线 视听 的 两 种 方法 。 本 节 先 来 介绍 html 标签 
中 的 播放 多 媒体 标签 <embed>。 

1. 基本 语法 

<embed src=url> 

url 为 音频 或 视频 文件 ， 其 路 径 可 以 是 相对 路 径 ， 也 可 以 是 绝对 路 径 。 

<embed> 可 以 用 来 播放 各 种 多 媒体 文件 ， 格 式 可 以 是 midi、wav、mp3 等 ， 当 下 主流 的 浏览 器 都 
支持 该 标签 。 

2. 属性 设置 

<embed> 标 签 有 很 多 属性 ， 常 用 的 属性 及 说 明 如 表 10.1 所 示 。 

表 10.1 <embed> 常 用 属性 及 说 明 


属 性 名 说 了 明 

该 属性 规定 音频 或 视频 文件 是 否 在 下 载 完 之 后 就 自动 播放 
音乐 下 载 完成 后 自动 播放 。false 下 载 完成 后 不 播放 

该 属性 规定 音频 或 视频 文件 是 否 循环 及 循环 次 数 

loop 属性 值 为 tue 时 ， 音 频 或 视频 文件 循环 ， 属 性 值 为 false 时 ， 

音频 或 视频 文件 不 循环 。 如 果 为 正 整数 ， 则 为 循环 次 数 

该 属性 规定 控制 面板 是 否 显示 ， 默 认 值 为 no 


举例 


autostart <embed src="1.mp3" autostart=true> 


<embed src="1.mp3" loop=true> 


hidden tre。 隐 藏 面板 ，no。 显 示 面 板 <embed src="1.mp3" hidden="yes"> 
该 属性 规定 音频 或 视频 文件 开始 播放 的 时 间 。 未 定义 则 从 文 | <embed src="1.mp3" 
starttime 
件 开头 播放 starttime="00:10"> 
| 该 属性 规定 音频 或 视频 文件 的 音量 大 小 。 未 定义 则 使 用 系统 pp wom ion 
Volume 本 身 的 设 定 ed src="1.mp3" volume= 


width 该 属性 规定 了 控制 面板 的 宽度 <embed src="1.Imp3" width="100"> 
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续 表 


说 ”了 明 
该 属性 规定 了 控制 面板 的 高 度 
该 属性 规定 音频 或 视频 文件 的 说 明文 字 


举例 


<embed src="1.mp3" height="200"> 


属 性 名 


<embed src="l.mp3" title="a good 


title 
Song"> 


除了 这 些 属性 外 ，<embed> 标 签 还 可 以 对 面板 的 外 观 及 其 背景 色 和 前 景色 进行 设置 ， 感 兴趣 的 读 
者 可 以 查找 相关 的 资料 。 


10.7.3 ”点 歌 模块 的 实现 过 程 


国 ”消息 管理 模块 使 用 的 数据 表 : tb_register、tb_video 

单 击 “ 点 歌 专区 ” 超 链 接 ， 首 先进 入 到 点 歌 专区 页 面 ， 页 面 中 的 音乐 试听 和 信息 查看 在 其 他 模块 
中 都 有 介绍 ， 这 里 主要 看 一 下 点 歌 功能 。 单 击 国 小 图 标 ， 将 进入 到 点 歌 信息 页 面 ， 点 歌 页 面 的 运行 结 
果 如 图 10.26 所 示 。 


4 点 歌 专区 Hh 

(I 本 -一 | 浊 入 点 到 区 | 2 人 [AT 
二 省 /ONEINE 影 视 365, (和 您 共 享 经 典 
i 二 vara 


您 有 4 条 短 消息 
用 户 名 : Teoft 【ap 矿泉 水 * 


点 歌 记录 详 单 一 点 歌 记录 详 单 


接收 人 , fo 


祝福 语 ， Pe 
习 


[点 家 ] 返 
图 10.26 点 歌 模块 的 运行 结果 


在 点 歌 信息 页 面 ， 需 要 用 户 添加 祝福 语 和 接收 入 名 称 。 点 歌 页 面 涉 及 的 HTML 表单 元 素 如 表 10.2 
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表 10.2 点 歌 页 面 中 的 主要 表单 元 素 


action="Register.asp?action=add" 


toname text | id="toname" size: | 
remark textarea | cols="40" rows="5" id="remark" | 
id Hidden | value="<?phpecho§ GET[id] ?>" | 
submit button | onClick="registerO" | 


button onClick="j 


在 点 歌 页 面 , 将 前 页 传 过 来 的 歌曲 id、 接 收入 、 祝 福 语 等 信息 组 成 insert 语句 存储 到 数据 库 中 , 不 
管 保存 成 功 与 否 ， 都 将 关闭 当前 子 窗口 ， 回 到 父 窗口 中 进行 操作 。 程 序 代 码 如 下 : 
例 程 04 ”代码 位 置 ， 光盘 \TM\10\online\give.php 


<?php 
session start(); // 开 启 session 支 持 
if($_POST[toname] > ""){ /如 果 接 收 人 不 为 空 
Sid=$_POST[id]: /歌曲 id 
Stoname=$_ POST[toname]: // 接 收入 姓名 
Sfrom=$_SESSION[name]:; // 发 送 人 姓名 
Sremark=$_POST[remark]: // 备 注 信息 


<!-- ”生成 insert 语 句 --> 
$sql="insert into tb_register Values(",".$id.",".$from.",".$toname.”,". Sremark.",".date("Y-m-d H:i:s").")"; 


S$rst = $conn->execute($sq]): /执行 SQL 语句 
?> 
<script language="javascript"> 
<2php 

if(!($rst 一 false)){ 
?> 

alert(" 点 歌 信息 保存 成 功 "): 
<?php 

}else{ 
?> 

alert(" 点 歌 失 败 "); 
<?php 
} 
?> 

top.window.close(): 
</script> 
<?php 
}; 
ye 
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10.8 ”在线 视听 模块 设计 


10.8.1 在 线 视听 模块 概述 


在 线 视听 主要 包括 视频 数据 在 线 观看 和 音频 数据 
在 线 收听 两 个 部 分 。 在 不 同 的 页 面 都 可 以 直接 进入 视 
听 页 面 ， 包 括 首页 、 搜 索 结 果 页 、 内 容 介绍 页 和 列表 
页 。 但 必须 是 登录 会 员 才 可 以 ， 游 客 是 没有 权限 的 。 
在 线 视 听 流程 图 如 图 10.27 所 示 。 


10.8.2 ”在线 视听 模块 技术 分 析 


在 10.7.2 节 中 学 习 了 如 何 使 用 <embed> 多 媒体 标 
签 来 播放 影音 文件 。 本 节 来 学 习 另 一 种 方法 : <object> 
标签 。 
使 用 <object> 标 签 来 定义 一 个 嵌入 的 对 象 。 该 对 象 10.27 在 线 试听 流程 图 
向 HTML 页 面 添加 多 媒体 。 此 元 素 运行 插入 到 html 页 面 中 的 对 象 的 数据 和 参数 ， 以 及 可 用 来 显示 和 操 
作 数 据 的 代码 。<object> 标 签 的 基本 格式 为 : 
<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" height="68" id="MediaPlayer1" width="460"> 
<param name="ShowStatusBar" value="-1"> 
<param name="Filename" value="<?php echo $_GET[id]?>"> 
</object> 
classid: 定义 能 入 Windows Registry 中 或 某 个 URL 中 的 类 的 ID 值 , 此 属性 可 用 来 制定 浏览 器 
中 包含 的 对 象 的 位 置 。 
回 param: 可 定义 用 于 对 象 的 run-time 设置 。 
<object> 是 微软 专门 为 正 浏览 器 打造 的 , 可 以 扩展 外 部 应 用 程序 及 插件 的 对 象 标签 , 它 和 <embed> 
的 不 同 之 处 在 于 : <object> 只 支持 以 正 技术 为 核心 的 浏览 器 系列 ， 对 其 他 的 浏览 则 无 效 。 而 且 如 果 要 
使 用 <object> 来 播放 多 媒体 ， 那 么 需要 安装 相应 的 播放 插件 ， 如 realplay、quicktime 等 。 


10.8.3 ”在线 视 听 模 块 的 实现 过 程 


国 。 在 线 视听 模块 使 用 的 数据 表 : tb_audio、tb_video 
会 员 可 以 在 首页 、 信 息 展 示 页 面 或 者 详细 信息 展示 页 面 单 击 “观看 ”或 “试听 ”按钮 进行 在 线 视 
听 操 作 。 页 面 运行 结果 如 图 10.28 所 示 。 
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图 10.28 在 线 视听 模块 的 运行 结果 


以 视频 详细 信息 展示 页 面 为 例 ， 用 户 可 以 单 击 型 图 标 ， 打 开 视 频 播放 页 面 (see.php) 进行 在 线 观 
看 。 视 频 播放 页 面 主要 是 根据 传递 的 参数 进行 数据 库 检 索 ， 并 将 对 应 数据 的 硬盘 存储 地 址 作为 多 媒体 
文件 的 引用 地 址 ， 当 页 面 加 载 完毕 后 ， 视 频数 据 将 自动 播放 ， 实 现在 线 观 看 功能 。 在 线 播放 页 面 的 主 
要 代码 如 下 : 


例 程 05 ”代码 位 置 : 光盘 \TM\10\online\see.php 
<?php 
session start(): // 开 启 session 支 持 
?> 
<body> 
<!-- 使 用 object 对 象 播放 视频 /音频 文件 -> 
<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" width="665" height="500" id="MediaPlayerl" > 
<!-- 设置 自动 播放 --> 
<param name="AutoStart" value="-1"> 
<L- 设置 状态 栏 -> 
<param name="ShowStatusBar" value="-1"> 
<!-- ”播放 文件 路 径 -> 
<param name="Filename" value="upfiles/audio/<?php echo $ GET[id]j: ?>"> 
</object> 
</body> 
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10.9 后 合 首 页 设计 
10.9.1 后 台 首 页 概述 


根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 , 本 系统 的 首页 面 中 要 显示 的 模块 主要 有 两 部 分 : 

(1) 网 站 左 侧 导航 栏 : 包括 各 个 管理 模块 及 分 类 。 

目录 管理 模块 : 主要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 

回 ”数据 管理 模块 : 主要 包括 视频 数据 管理 和 音频 数据 管理 两 个 部 分 。 

会 员 管 理 模块 : 包括 会 员 等 级 管理 、 会 员 信息 冻结 、 会 员 信息 删除 3 个 部 分 。 

回 上传 信 息 管 理 模块 : 主要 用 于 浏览 和 查询 用 户 上 传 操 作 的 详细 内 容 。 

加 ”管理 员 设 置 模块 ， 主 要 包括 管理 员 信息 添加 、 删 除 、 冻 结 3 个 部 分 。 

(2) 网 站 主 显示 区 : 显示 各 个 模块 的 操作 及 结果 。 

下 面 看 一 下 本 案例 中 提供 的 后 台 首 页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\10\online\admin\ 
index.php， 如 图 10.29 所 示 。 


i 
A 所 


CNWINE 影 视 365,( 和 全 共享 经 典 
~ ”i 


图 10.29 ”online 影视 365 网 后 台 首 页 
10.9.2 后台 首页 技术 分 析 


本 系统 的 后 台 首 页 采用 的 布局 结构 为 二 分 栏 布 局 。 左 侧 导 航 栏 清楚 、 明 白地 显示 了 后 台 管 理 员 所 
能 使 用 到 的 功能 。 当 单 击 任意 功能 按钮 时 ， 在 主 显示 区 显示 对 应 的 操作 界面 和 该 功能 模块 下 的 子 功能 。 
页 面 简练 、 结 构 清晰 ， 用 户 浏览 方便 ， 二 分 栏 的 主要 特点 被 表现 得 淋漓 尽 致 。 关 于 二 分 栏 的 详细 介绍 ， 
可 查看 3.6.3 节 和 4.7.1 节 的 相关 内 容 。 
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10.9.3 后台 首 页 的 实现 过 程 


国 后台 首页 使 用 的 数据 表 : tb_manager 

下 面 来 看 一 下 后 台 首 页 的 实现 过 程 。 在 首页 代码 中 , 对 每 项 功能 权限 的 判断 都 是 在 left.php 中 完成 
的 ， 只 要 在 左 侧 导 航 栏 的 位 置 载 入 leftphp 即 可 。 在 主 显示 区 ， 根 据 参数 返回 的 不 同 值 ， 显 示 不 同 的 功 
E 界 面 。 程 序 的 关键 代码 如 下 : 

例 程 06 ”代码 位 置 光盘 \TM\10\online\admin\main php 


<?php 
session start(); // 开 启 session 支 持 
include "inc/chec.php": // 载 入 权限 检查 文件 
include "conn/conn.php"; // 载 入 数据 库 链接 文件 
?> 
<link href="css/style.css" rel="stylesheet"/> // 引 入 css 样 式 文件 
<script src="js/admin js.js" language="javascript"></script> /1/ 引 入 js 脚本 文件 
<center> 


<?php 

局 ” 载 入 网 站 左 侧 导 航 栏 */ 
include "left.php": 

?> 


<div style="height:35px:;">&nbsp;</div> 
<?php 
必 ”根据 不 同 的 参数 ， 显 示 不 同 的 功能 界面 */ 
iflisset($_GET[action])){ 
Switch ($_GET[action]){ 
case "audioList": // 视 频 目 录 名 称 添加 
include "a_list.php"; 
break:; 
case "videoList": // 音 频 目 录 添加 
include "v_listphp": 
break: 
case "audio": // 视 频数 据 添加 
include "audio.php": 
break:; 
case "video": /音频 数据 添加 
include "Video.php": 
break: 
case "grade": // 会 员 等 级 设置 
include "grade.php": 
break:; 
case "member": // 会 员 数 据 设置 
include "member.php": 
break: 
case "log": /上 传 日 志 管理 
include "log.php": 
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break: 
case "manager": // 管 理 员 设置 
include "manager.php": 
break: 
了 六 
本 </td></t> 
<tr><td height="15" bgcolor="#f0f0f0">&nbsp:;</td></tr> 
</table> 
</td></t> 
</table> 
</center> 


10.10 目录 管理 模块 设计 


10.10.1 目录 管理 模块 概述 


目录 管理 模块 主要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 管 理 员 可 以 通过 后 台 管理 导航 进 
入 对 应 的 目录 管理 页 面 。 

如 果 管 理 员 进入 了 视频 目录 管理 页 面 (audiolistphp) ， 那 么 在 该 页 面 内 可 以 打开 目录 添加 页 面 进 
行 目录 添加 操作 ， 也 可 以 删除 相应 的 目录 。 鉴 于 目录 信息 比较 简短 ， 本 例 中 没有 提供 目录 修改 的 功能 。 
目录 管理 模块 的 用 例 图 如 图 10.30 所 示 。 


一 > 删除 音频 目录 ) < 一 一 一 一 > (添加 视频 目录 ) < 一 一 
一 > 添加 音频 目录 ) < 一 一 一 一 删除 视频 目录 ) < 一 一 
音频 目录 管理 员 超级 用 户 视频 目录 管理 员 


10.30 ”视频 目录 管理 用 例 图 


[0 说 明 : 由 于 视频 目录 管理 和 音频 目录 管理 实现 的 方法 基本 类 似 ， 因 此 ， 本 章 重点 讲解 视频 目录 管 
理 模块 实现 的 方法 ， 关 于 音频 目录 管理 模块 的 实现 方法 请 参见 本 书 附 赠 源码 光盘 。 


10.10.2 目录 管理 模块 技术 分 析 


在 本 系统 中 ， 有 个 功能 经 常 要 被 用 到 ， 就 是 在 添加 目录 、 和 歌曲 时 ， 经 常会 碰 到 重 名 的 问题 。 对 于 
这 个 问题 ， 可 以 写 一 个 自 定义 函数 ， 并 将 它 放 到 单独 的 一 个 文件 中 ， 方 便 其 他 页 面 调用 ， 如 本 系统 中 
的 /inc/func.php 就 是 专门 用 来 存储 自 定义 函数 的 文件 。 

下 面 来 看 一 下 自 定义 函数 is_chk0， 其 代码 如 下 : 

例 程 07 ”代码 位 置 : 光盘 \TM\10\online\admin\inc\func.php 


// 判 断 目录 名 是 否 重复 
//Sf fields: 字段 名 
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//Stablename: 数据 表 名 
/Sf str: 要 查找 的 字段 
function is_chk($f fields.$tablename.$f str){ 


S$conn = &ADONewConnection(mysql): // 建 立 mysql 连 接 
Sconn->PConnect("localhost","root"."root"."db_online"): // 连 接 “db_online” 数 据 库 
Sis_chk = tmue: // 声 明 一 个 boolean 型 变量 
Sis_sqlstr = "select $f fields from $tablename": // 查 找 字段 

Sis_Ist = $conn->execute($is_sqlstr): 

while(!$is rst->EOF){ 1/ 循环 输出 记录 集 


/*#### 站 下 #### 一 一 -对比 字 段 中 的 值 。 站 # 汪 本 二/ 
if($f str 一 Sis rst->fields[0]){ 
让 ”如 果 字 有 段 在 数据 库 中 找到 ， 说 明 重 名 了 ，is_chk 设 置 成 false */ 


S$is_chk = false; 
break:; 
1 
$is_Tst->MoveNextO: /将 指针 指向 下 一 条 记录 
由 
Teturn S$is_chk: // 返 回 变量 $is_chk 


} 
10.10.3 ”目录 管理 的 实现 过 程 


国 目录 管理 使 用 的 数据 表 : tb_audiolist、tb_videolist 

单 击 左 侧 导 航 栏 中 的 “视频 目录 管理 ” 超 链接 ， 可 以 打开 视频 目录 管理 页 面 ， 在 管理 页 面 中 包含 
一 个 “目录 添加 ” 超 链接 、 所 有 的 一 、 二 级 目录 信息 和 对 应 的 “删除 ”按钮 。 视 频 管理 页 面 的 运行 结 
果 如 图 10.31 所 示 。 


y i yen 
rs 的 


视频 目录 管理 
目录 寺 加 
m Su 名 称 丸 闻名 称 抬 作 
21 2 在 淮 剧 入 电视 岗 队 
六 六 大 车 可 革 电视 人 
-Nt: 电影 MS 
1 1 电视 一 了 
iT 2 豆 则 电影 及 
18 2 怠 估 电影 各 院 
1 2 和 和 eR 二 
22 2 其 她 刚 襟 电视 出 除 
中 绊 欧美 出 入 电视 而 际 
2 2 [5 电视 bi 


图 10.31 视频 目录 管理 的 运行 结果 
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视频 目录 管理 页 面 的 代码 很 简单 : 使 用 while 循环 将 数据 库 中 的 记录 全 部 显示 出 来 ,并 在 每 输出 一 
条 记录 后 ， 添 加 一 个 “删除 ”操作 。 目 录 管理 页 面 的 实现 代码 如 下 : 


例 程 08 ”代码 位 置 : 光盘 \TM\10\online\admin\a_ list.php 


<?php 
session start(); // 开 启 session 支 持 
include "conn/conn.php"; // 载 入 数据 库 链 接 文 件 
include "inc/chec.php": // 载 入 权限 检查 文件 


必 ”输出 视频 目录 数据 表 中 的 全 部 数据 */ 
$l sqlstr = "select * from tb_audiolist"; 
Sl rst = $conn->execute($] sqlstr):; 


?> 


<t> 
作 目录 添加 */ 
<td height="10" colspan="5" align="right" valign="middle"><a href="#" 
onclick="javacript: Wopen=open('operation.php?action=audiolist',' 添 加 目录 ','height=500,width=665,scrollbars=n0');"> 
目录 添加 </a></td> 
</t> 
<!-- 显示 目录 相关 信息 的 字段 名 --> 
<t> 
<td height="30" align="center" valign="middle">ID</td> 
<td height="30" align="center" valign="middle"> 等 级 </td> 
<td height="30" align="center" valign="middle"> 名 称 </td> 
<td height="30" align="center" valign="middle"> 父 级 名 称 </td> 
<td height="30" align="center" valign="middle"> 操 作 </td> 
</t> 
<?php 
while(!S$] rst->EOF){ 
> 
<t> 
<!-- 输出 目录 id --> 
<td height="18" align="center" valign="middle"><?php echo $]_rst->fields[0]: ?></td> 
<!-- 输出 目录 等 级 --> 
<td height="18" align="center" valign="middle"><?php echo $l_ rst->fields[1]: ?></td> 
<!-- 输出 目录 名 称 -> 
<td height="18" align="center" valign="middle"><?php echo $]_rst->fields[2]: ?></td> 
<!-- 输出 目录 父 级 名 称 --> 
<td height="18" align="center" valign="middle"><?php echo $]_rst->fields[3] ?></td> 
<!-- 输出 “删除 ” 超 链接 -> 
<td height="18" align="center" valign="middle"><a 
href="del list_chk.php?action=audiolist&id=<?php echo $1_rst->fields[0]: ?>" onclick="return del chkO:"> 删 除 </a> 
</td> 
</t> 


<2php 
上 # 指针 下 移 一 条 记录 。 */ 
Sl_rst->MoveNext(): 
} 


> 
</table></td> 
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</t> 
</table></td> 
</u> 
</table> 


10.10.4 目录 添加 的 实现 过 程 
国 ”目录 添加 使 用 的 数据 表 : tb_audiolist、tb_videolist 


单 击 视频 目录 页 面 中 的 “目录 添加 ”按钮 ， 将 打开 目录 添加 页 面 。 目 录 添 加 页 面 的 运行 结果 如 
图 10.32 所 示 。 


视频 目录 名 称 洪 加 


目录 和 名称:FR 

GEL EHE CEE| 

交 刀 名称: [ 虽 有 习 
区 到 [区 可 


图 10.32 视频 添加 目录 的 运行 结果 
视频 目录 添加 页 面 (audiolist.php) 中 主要 包含 一 个 添加 目录 表单 , 表单 的 主要 元 素 如 表 10.3 所 示 。 
表 10.3 ”视频 目录 添加 页 面 涉及 到 的 HTML 表单 元 素 


名 称 类 型 重要 属性 含 义 
list form method="post" action="audiolist_chk.php" 视频 目录 添加 表单 
name="grade" OnChange="check()" 
grade select <option value="1" selected> 一 级 目录 </option> 目录 等 级 
<option value="2"> 二 级 目录 </option> 
<?php 


$l sqlstr = "select * from tb_audiolist where grade ="1"™; 
$l rst = $conn->execute($] sqlstr); 

while(!S$] rst>EOF){ ?> 

father select A > A 父 级 目录 名 称 
<option value="<?php echo $l rst->fields[2]: ?>"><?php echo 
$l_rst->fields[2]: ?></option> 


<2php $L Tst>MoveNext(0: 
}7> 
names text type="text" id= "names” 目录 名 称 
Submit submit class="submit" value=" 添 ”加 " onclick="return n_chk0:" “添加 ”按钮 
class="submit" value=" 返 ” 回 " 
Submit2 button a _ ， “返回 ”按钮 


474 


第 10 章 online 影视 365 网 ( Apache+PHP+ADODB+phpMyAdmin+MySQL 5.0 实现 ) 


当 管 理 员 添加 信息 提交 后 ， 系 统 转 到 处 理 页 中 (audiolist chk.php) 进行 添加 操作 。 首 先 判断 输入 
的 目录 名 称 是 否 与 已 存在 的 名 称 重复 ， 如 果 重 复 ， 回 到 上 一 步 ， 如 果 不 重复 ， 则 向 数据 表 中 添加 新 记 
录 。 视 频 目录 添加 的 程序 代码 如 下 : 


例 程 09 ”代码 位 置 : 光盘 \TM\10\online\admin\audiolist_chk php 


<?php 
session start(); // 开 启 session 支 持 
include "inc/chec.php": // 载 入 权限 检查 文件 
include "conn/conn php"; // 载 入 数据 库 连 接 文 件 
include "inc/func.php"; // 载 入 自 定义 函数 文件 


入 使 用 自 定义 函数 is_chk0， 判 断 用 户 名 是 否 重复 */ 
iflis_chk(“name","tb_audiolist",$_ POST[names]) 一 false){ 
echo "<script>alert(' 名 称 重复 "):history.go(-1):</script>": /如 果 重 复 ， 退 到 上 一 步 操作 
exit(); 


; 
请 生成 insert 插 入 语句 */ 
S$a_sqlstr = "insert into tb_audiolist (grade,name,father,userName,issueDate) 
values('$ POST[gradel','$ POST[namesl'$ POSTI[fatherl''$ SESSION[adminl',".date("Y-m-d H:i:s").")"; 
入 ”判断 返回 结果 */ 
if($conn->execute($a_sqlstr) 一 false) 
族 ”如果 添加 失败 ， 返 回 上 一 步 */ 
echo "<script>alert( 添 加 失败 "):history.go(-1):</script>": 
else 
店 ”如 果 添 加 成 功 ， 则 刷新 父 窗口 ， 关 闭 子 窗口 */ 
echo "<script>top.opener.location.reload():alert( 添 加 成 功 ):window.closeO:</script>"; 
?> 


10.10.5 ”目录 删除 的 实现 过 程 


国 ”目录 删除 使 用 的 数据 表 : tb_audiolist、tb_videolist 
视频 目录 的 删除 功能 是 根据 传递 的 目录 id 号 将 数据 库 中 对 应 的 记录 找到 , 并 删除 , 程序 代码 如 下 : 
例 程 10 ”代码 位 置 : 光盘 \TM\1Ovonlinevadminvdel list_chk.php 


<?php 

session start(): // 开 启 session 支 持 
include "inc/chec.php": // 载 入 权限 检查 文件 
include "conn/conn.php": // 载 入 数据 库 链 接 文件 
if($_GET[action] =— "audiolist") // 如 果 action 等 于 audiolist 

S$t_name = "tb_audiolist"; // 要 访问 的 数据 表 就 为 tb_audiolist 
else if($_GET[action] 一 "videolist") // 如 果 action 等 于 videolist 

S$t_name = "tb_videolist": // 那 么 数据 库 表 就 为 tb_videolist 
S$sqlstr = "delete from ".$t_name." where id =".$_GETI[id]: /根据 id 号 删除 对 应 记录 


睛 “如果 删除 成 功 ， 则 返回 到 当前 页 ， 否 者 ， 退 到 上 一 步 */ 
if($rst = $conn->execute($sqlstr) == false){ 
echo "<script>alert( 删 除 错误 ! ".$sqlstr."):history.go(-1):</script>": 
}else 
echo "<script>alert( 删 除 成 功 "):location=".$_SERVER['HTTP_REFERER'].":</script>"; 
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10.11 数据 管理 模块 设计 


10.11.1 数据 管理 模块 概述 


数据 管理 模块 主要 包括 视频 数据 管理 和 音频 数据 管理 两 个 部 分 。 管 理 员 可 以 通过 后 台 管 理 导航 进 
入 对 应 的 数据 管理 页 面 ， 如 视频 数据 管理 页 面 (audiophp) 。 在 数据 管理 页 面 中 ， 用 户 可 以 打开 数据 
添加 页 面 或 删除 对 应 的 数据 。 数 据 管理 模块 用 例 图 如 图 10.33 所 示 。 


2 


音频 目录 管理 员 超级 用 户 视频 目录 管理 员 


GD Ga 


10.33 ”数据 管理 模块 用 例 图 


人 说 明 : 由 于 视频 数据 管理 和 音频 数据 管理 实现 的 方法 基本 类 似 ， 因 此 ， 本 章 重 点 讲解 视频 数据 管 
理 模块 实现 的 方法 ， 关 于 音频 数据 管理 模块 的 实现 方法 请 参见 本 书 附 赠 源码 光盘 。 


10.11.2 ”数据 管理 模块 技术 分 析 


在 本 模块 中 , 主要 运用 的 就 是 文件 上 传 技术 。 关 于 文件 上 传 方面 的 知识 , 请 参见 本 书 8.7 节 的 内 容 ， 
里 面 有 对 预定 义 变量 $_ FILES 的 详尽 介绍 和 如 何 判断 图 片 类 型 和 大 小 的 相关 方法 。 
在 这 个 模块 中 ， 略 有 不 同 的 是 使 用 了 一 个 自 定义 函数 来 判断 文件 名 后 级 。 下 面 来 看 看 这 个 函数 的 
实现 过 程 ， 代 码 如 下 : 
例 程 11 代码 位 置 : 光盘 \TM\1Ovonlinevadminincfunc php 
// 判 断 文件 后 级 
/1/$f_type: 允许 文件 的 后 级 类 型 
JSf upfiles: 上 传 文件 名 
function f postfix(Sf type.Sf upfiles){ 


Sis_pass = false: // 设 置 一 个 boolean 型 变量 $is_pass 
oo Stmp_upfiles = split(™\.",Sf upfiles): /|/ 拆 分 上 传 文件 名 ， 存 储 到 变量 Stmp_upfiles 中 
@ Stmp_ fix = Stmp_upfiles[count(Stmp_upfiles) - 1]: // 取 得 上 传 文件 名 后 缀 

for($num = 0: Snum < count(Sf type):SnumtHH{ /使 用 for 循 环 ， 输 出 允许 使 用 的 后 缀 类 型 
目 这 strtolower(Stmp_tix) 一 $f type[Snum]) // 如 果 上 传 文件 后 缀 包括 在 其 中 ， 说 明 类 型 相符 
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Sis_pass =true:; JSis_pass 变 量 赋值 为 tue 
} 
Teturn $is_pass: // 返 回 变量 $is_pass 


< 代码 由 二 

@ split("\.",$f upfiles): splitO 函 数 为 字符 分 割 函 数 ， 它 有 两 个 参数 : 分 隔 符 、 要 分 割 字符 囊 。 返 回 结果 为 分 割 后 的 
字符 串 数组 。 例 如 ，$tmp=split("c","placepublicslice")， 那 么 Stmp 值 就 为 Stmp[0]="plac"、S$tmp[1]="epubli"、S$tmp[3]="sli"、 
Stmp[4]="e". 

@ count($tmp upfiles): count0 函 数 的 作用 是 取得 数组 个 数 。 

目 strtolower(Stmp tix): strtolower0 函 数 返 回 参 数字 符 串 的 小 写 格式 ， 如 strtolower("AbC")， 输 出 等 于 abc。 


人 说 明 : 尽 可 能 地 使 用 PHP 函数 库 的 函数 。 因 为 自己 写 的 函数 ， 无 论 是 从 算法 设计 ， 还 是 从 功能 实现 
上 ， 都 无 法 和 PHP 自 带 的 函数 相 比较 。 不 要 试 着 去 找寻 它们 的 错误 , 这 是 非常 不 明智 的 做 法 。 


10.11.3 数据 添加 的 实现 过 程 


国 ”数据 添加 使 用 的 数据 表 : tb audio、tb_video 

管理 员 可 以 通过 单 击 视频 数据 管理 页 面 的 “数据 添加 ”按钮 打开 数据 添加 页 面 (audiophp) ， 在 
数据 添加 页 面 填写 了 表单 之 后 ， 将 表单 提交 给 本 页 ， 本 页 会 根据 提交 的 表单 数据 在 数据 库 中 添加 对 应 
的 记录 。 数 据 添加 页 面 的 运行 结果 如 图 10.34 所 示 。 


视频 数据 深 加 


JU 下 


My ee EL 全 大 room 
Decrements aa 


发 行 时 间 : Euosorol 加。 


新 虽 : @ 是 C 再 * 


区 大片 ， 禄 什 得 一 奢 a 
简要 介绍 : 
习 


[ER 


10.34 ”视频 数据 添加 页 面 的 运行 结果 
在 视频 数据 添加 页 面 中 涉及 到 的 HTML 表单 元 素 如 表 10.4 所 示 。 
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表 10.4 视频 添加 页 面 涉及 到 的 HTML 表单 元 素 


名 称 类 型 重要 属性 含义 
Lt "uA name="list" He Oe action="dataadd chkphp" 表单 
enctype="multipart/form-data” 
Names text id="names" size="30" 数据 名 称 
Picture text id="picture" size="15" 图 片 名 称 
Address text id="address" size="15" 数据 位 置 
<option value=" 一 级 "> 一 级 </option> 
<option valu "> 二 级 </option> 
Grade Select <option value=" 三 级 "> 三 级 </option> 数据 等 级 
<option value=" 无 限制 级 " selected="selected"> 无 限制 级 </option> 
<option value=" 禁 片 "> 禁片 </option> 
Publisher text id="publisher" size="30" 发 行商 
Actor text id="actor" size="30" 主要 演员 
Director text id="director" size="30" 导演 
Maker text id="marker" size="30" 制 片 人 
<input type="radio" name="language" value=" 中 文 " checked> 中 文 
<input type="radio" nanx 英文 "> 英文 
<input type="radio" name="language" value=" 韩 语 "> 韩语 
Language radio <input type="radio" name="language" value=" 日 语 "> 日 语 语种 
<input type="radio" name="language" valu 语 "> 德语 
<input type="radio" name="language" value=" 法 语 "> 法 语 * 
<ltd> 
<?php S$a_sqlstr="select * from tb_videoList where grade="2"; 
Sa_Ist = $conn->execute($a_sqlstr); 
while(!S$a_rst->EOF){ 
k 
Style select <option value="<?php echo 类 别 
Sa_rst->fields[2]: ?> "><?php echo $a_rst->fields[2]: ?></option> 
<?php Sa_rst->movenext(): 
} 
?> 
<?php 
S$t_sql = "select * from tb_audiolist where grade = 
S$t_Ist = $conn->execute($t_sq]): 
while(!S$t_rst->EOF){ 
?> 
Types select <option value="<?php echo S$t_rst->fields[2]: ?>"><?php echo 类 型 
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St_rst->fields[2]: ?></option> 
<?php 
S$t_Ist->movenext(): 


} 


?> 
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续 表 


名 称 | 类 型 含义 
From text | id="from" size="30" 发 行 国家 
publishtime text | loadCalendar(listpublishtime): 发 行 时 间 

<input name="news" type="radio" value="1" checked> 是 
We a <input name="news" type="radio" value="0"> 否 册 才 数 
remark textarea | class="submit" onclick="return a_chkO:” 简要 介绍 

Submit button | class="submit" onclick="retum a_chkO:” “添加 ”按钮 
Submit2 button OnClick="javascripttop.window.close0"> “返回 ”按钮 


视频 管理 处 理 页 (dataadd_chk) 主要 分 为 3 部 分 。 图 片 文件 格式 及 图 片 大 小 判断 、 视 /音频 格式 及 
文件 大 小 判断 ， 信 息 都 准确 无 误 后 ， 则 向 数据 库 中 添加 新 记录 。 处 理 页 处 理 的 程序 代码 如 下 : 
例 程 12 ”代码 位 置 ; 光盘 \TM\10\onlinevadminvdataadd chkphp 


<?php 
session start(); /开启 session 支 持 
include "inc/chec.php"; // 载 入 权限 检查 文件 
include "conn/conn.php"; // 载 入 数据 库 链 接 文件 
include "inc/func.php"; // 载 入 自 定义 函数 文件 
Sp_type = array("jpe","jpeg","bmp","gif"): // 允 许 上 传 的 图 片 文件 格式 
Sf type = array("avi", "rm","rmvb","wav","mp3"."mpg"): // 允 许 上 传 的 视频 、 音 频 格 式 
S$audio_path = "../upfiles\audio":; // 视 频 上 传 位 置 
Svideo_path = "../upfiles\\video": // 音 频 上 传 位 置 
Spicture_path =""; // 图 片上 传 路 径 
Sfile path=""; // 文 件 上 传 路 径 


在 图 片 类 型 判断 中 ， 首 先 判断 图 片 大 小 是 否 在 允许 范围 之 内 ,接着 使 用 自 定义 函数 f_postfix0 来 判 
断 图片 后 缀 是 否 符合 要 求 , 如 果 没 有 问题 则 根据 time0 函 数 ， 生 成 图 片 名 , 并 保存 到 变量 $picture_path 
中 ， 代 码 如 下 : 
例 程 13 ”代码 位 置 : 光盘 \TM\10\online\admin\dataadd_chk.php 
谨 判断 图 片 大 小 ， 是 否 大 于 0k， 小 于 700k */ 


0 if($_FILES[picture][size] > 0 and $ FILES[picture][size] < 700000){ 
店 ”如 果 图 片 格式 正确 */ 

© if((S$postf =f postfix(Sp_type,s_FILES[picture][name])) != false){ 
人 tt so 

© Spicture_path = time0.". 


谨 ”如 果 action 值 为 a 时 ， 局 明了 乱 为 视频 图 片 Ee 
if($ POST[action] 一 "a"){ 
请 “如果 生成 的 临时 文件 正确 ， 则 使 用 move_uploaded file0 函数 上 传 */ 
这 $_FILES[picture][tmp name]) 
@ move_uploaded file($ FILES[picture][tmp_name],Saudio path."\".Spicture_path); 

else{ 
echo "<script>alert( 上 传 图 片 失败 ! '"):history.go(-1):</script>"; 
exitO: 


} 
店 ”如果 上 传 图 片 为 音频 图 片 */ 
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jelse if($ POST[action] — "v"){ 
这 $_FILES[picture][tmp_name]) 
Imove Uploaded file($ FILES[picture][tmp name],$video path."\".$picture_path); 
else{ 
echo "<script>alert( 上 传 图 片 失败 ! "):history.go(-1):</script>": 
exit(); 
} 
} 
}else{ 
echo "<script>alert( 上 传 图 片 格式 错误 ! "):history.go(-1):</script>"; 
exit(): 


必 ”如 果 图 片 大 于 700KB, 说 明 图 片 太 大 ， 超 出 要 求 ， 返 回 上 一 步 */ 
Yelse if($ FILES[picture][size] > 700000){ 
echo "<script>alert( 上 传 图 片 大 小 超出 范围 ! "):history.go(-1):</scrip>"; 
exit(); 


} 


} 
else{ 
Spicture = ""; 


/这 率 素 素 率 闪 率 素 闪 率 素 素 率 六 素 闵 六 六 闪闪 率 站 闪 率 闪 站 素 半 闲 让 / 


人 代码 贴 二 


@ 使 用 预定 义 变量 的 $_FILES[name][size] 属 性 判断 上 传 文件 的 大 小 。 如 果 文件 大 于 pho ini 中 设置 的 上 传 文件 的 最 


大 值 ， 那 么 size 将 永远 返回 0， 而 不 是 文件 的 实际 大 小 。 


@ 使 用 自 定义 函数 fpostfix() 判 断 上 传 文件 的 后 组 。 
@ 返回 当前 UNIX 的 时 间 蕉 . 
@ 使 用 move uploaded file() 上 传 函 数 要 注意 ; 它 的 第 1 个 参数 是 上 传 到 服务 器 中 的 临时 文件 名 ， 而 不 是 文件 原始 


名 称 ， 是 用 户 自 定义 的 名 称 。 要 取得 临时 文件 名 使 用 $_FILES[name][tmp_name]. 


对 上 传 文件 的 判断 和 上 传 图 片 的 流程 基本 相同 ， 也 是 先 判断 文件 大 小 是 否 符合 上 传 文件 的 范围 要 
求 ， 接 着 判断 上 传 文件 后 级 ， 最 后 使 用 time0O 函 数 生成 文件 名 ， 保 存 到 变量 Saddress path 中 。 下 面 是 判 


断 视频 文件 的 流程 ， 代 码 如 下 : 
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例 程 14 ”代码 位 置 : 光盘 \TM\1O\onlinevadminvdataadd_ chkphp 


上 请 判断 上 传 文件 类 型 与 大 小 */ 


if($_FILES[address][size] > 0){ // 判 断 上 传 文件 大 小 
这 $ POST[action] 一 "a"){ /判断 是 视频 文件 还 是 音频 文件 
if($_FILES[address][size] < 300000000){ // 判 断 文 件 是 否 超出 限定 大 小 
if(($postf =£ postfix($f type.$_FILES[address][name])) 二 false){ 
Sfile_path = time().".".$postf: // 生 成 新 的 文件 名 
if($_FILES[address][tmp_name]) // 如 果 临 时 文件 正确 


刻 使 用 move_uploaded_file0 函数 上 传 文件 
move_uploaded file($ FILES[address][tmp_name].$audio path."\".$file pathb): 
else{ 
echo "<script>alert(' 上 传 文件 错误 ! "):history.go(-1):</script>"; 
exitO: 


}else{ 
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echo "<script>alert( 上 传 文件 格式 错误 ! '):history.back(-1);</script>"; 
exit(); 

}}else{ 

echo "<script>alert( 上 传 文件 大 小 错误 ! "):history.go(-1):</script>": 

exit(); 


入 
全” 如 果 是 音频 文件 */ 
else if($ POST[action] 一 "v"){ 
店 ”判断 文件 大 小 */ 
if($_FILES[address][size] < 10000000){ 
局 ” 判 断 文 件 后 缀 */ 
这 ($postf=f postfix($f type,$_FILES[address][name])) 二 false){ 
S$file path = time().".".$postf: 
让 ”判断 临时 文件 是 否 生成 */ 
if($_FILES[address][tmp_name]) 
让 ”使 用 move_uploaded_file0 函数 上 传 文件 */ 
move_uploaded file($ FILES[address][tmp_name],$video_path."\".$file_path); 


else{ 
echo "<script>alert( 上 传 文件 错误 ! "):history.go(-1);</script>"; 
exit():}}else{ 
echo "<script>alert( 上传 文 件 格 式 错误 ! "):history.back(-1):</script>"; 
exit();}}else{ 
echo "<script>alert( 上传 文 件 大 小 错误 ! "):history.go(-1):</script>"; 
exitO:}} 
}else{ 
echo "<script>alert( 没 有 上 传 文件 或 文件 大 于 300M"):history.go(-1):</script>"; 
exit(); 
} 


如 果 上 传 图 片 和 上 传 的 文件 都 没有 问题 ， 则 对 表单 的 其 他 信息 进行 处 理 ， 生 成 insert 插入 语句 ， 添 
加 新 记录 。 程 序 代码 如 下 : 


例 程 15 ”代码 位 置 : 光盘 \TM\10\online\admin\dataadd_chk.php 
族 ”视频 文件 和 音频 文件 相同 的 信息 */ 


Snames = $_POST[names]: // 视 频 名 称 
Serade =$_POST[grade]: /级 别 
S$sizes =$_FILES[address][size]: /大 小 
Spublisher = $_POST[publisher]: /发 行商 
Sactor = $_POST[actor]: /演员 
Slanguage = $_ POST[language]: /语言 
S$style =$_POST[style]: /类 别 
Stypes = $_POST[types]: // 类 型 
Sfroms = $_ POST[from]: // 发 行 国家 
Spublishtime = $_POST[publishtime]: // 首 映 时 间 
Snews = $_POST[news]: // 是 否 为 新 片 
Sremark = $_POST[remark]: // 备 注 


这 $_POST[action] 一 "a"){ 
/* 确认 父 级 目录 */ 
记 “表单 值 ， 视 频 文件 特有 的 属性 */ 
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Sdirector = $_POST[director]: // 提 取 导 演 

Smarker = $_POST[marker]: 1/ 制 片 人 

族 ”生成 insert 语 句 */ 

$a_sqlstr = "insert into tb_audio 
(name,picture,sizes.erade.publisher.actor.director.marker,languages.type.style.froms.publishtime,bool.remark.property.address.u 
sername,issueDate) 
values('$names','$picture_path','$sizes','$erade','$publisher’,'$actor','S$director'.'$marker','$language','$types','$style','$from','$publ 
ishtime','$news','remark'", 管 理 员 ','$file path','$_SESSION[admin],".date("Y-m-d H:i:s").")":; 


中 

else if($ POST[action] =— "v"){ // 如 果 上 传 文件 为 音频 文件 
$actortype = $_POST[actortype]: // 歌 曲 类 型 
$ci=$ POST[ci]; // 作 词 
$qu=$_POST[qu]: /作曲 


久生 成 insert 语 名 */ 
$a_sqlstr = "insert into tb_video 
(name,picture,actor,ci,qu,actortype,type,style,publisher,froms,sizes,languages.publishTime,remark,property,address,u 
serName,issueDate) 
values('$names','$picture_path','$actor','Sci','$qu','$actortype','$types','S$style','$publisher’,'$froms','$sizes','$language','$publishti 
me','$remark'", 管 理 员 ','$file_path''$_SESSION[admin],".date("Y-m-d H:i:s").")": 
}else{ 
echo "<script>alert( 错 误 ');window.close(:</script>"; 
exit(); 
让” 上传 图 片 和 文件 */ 
/本 来 束 束 束 束 事 束 束 束 来 事 束 束 束 束 可 来 束 束 束 宗 束 束 束 可 来 人 
Sa_Ist = $conn->execute($a_sqlstr):; 
这 !($a_rst 一 false)) 
echo "<script>top.opener.location.reload():alert( 添 加 成 功 ):window.closeO:</script>"; 
else 
echo "<script>alert( 添 加 失败 "):history.go(-1):</script>": 
?> 


10.11.4 ”数据 删除 的 实现 过 程 


国 ”数据 删除 使 用 的 数据 表 : tb_audio、tb_video 
视频 数据 删除 主要 是 根据 传递 的 文件 id 值 ， 将 数据 库 中 的 记录 删除 ， 同 时 ， 使 用 unlink 函数 删除 
服务 器 端的 真实 数据 。 程 序 代码 如 下 : 


例 程 16 代码 位 置 光盘 \TM\10\online\admin\del video_chk php 


<?php 
session start():; // 开 启 session 支 持 
include "conn/conn.php": // 载 入 数据 库 链 接 文 件 
include "inc/chec.php": // 载 入 权限 判断 文件 
S$file path = "../upfiles/video/": // 生 成 文件 路 径 


$s_sqlstr = "select * from tb_video where id =".$ GET[id]: // 生 成 SQL 语 句 
S$s_Ist = $conn->execute($s_sqlstr); 
这 !($s rst — false)){ 

让 ”删除 相应 的 图 片 文件 和 音频 、 视 频 文件 */ 

if(unlink($file path.$s_rst->fields[16]) and unlink($file path.$s rst->fields[2])){ 
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卢 ”生成 删除 语句 */ 
$d_sqlstr = "delete fom tb_video where id=".$ GET[id]: 
卢 执行 删除 语句 */ 
$d_rst= $conn->execute($d sqlstD: 
这 !(Sd rst — false){ 
echo "<script>alert(' 删 除 成 功 "):location='main.php?action=video':</script>"; 
}else{ 
echo "<script>alert( 删 除 失败 "):history.go(-1):</script>": 
}}}else 
echo "<script>alert( 删 除 失 败 人):history.go(-1):</script>"; 


10.12 ”管理 员 设 置 模块 设计 
10.12.1 管理 员 设 置 模块 概述 


管理 员 设置 模块 主要 包括 管理 员 信息 添加 、 信 息 删 除 及 信息 冻结 3 个 部 分 。 管 理 员 可 以 通过 后 台 
管理 功能 导航 进入 管理 员 设置 页 面 (memberphp) 。 在 设置 页 面 中 ， 管 理 员 可 以 进行 管理 员 信息 的 添 
加 操作 ， 或 进行 会 员 信息 的 冻结 或 删除 等 操作 。 管 理 员 设 置 用 例 图 如 图 10.35 所 示 。 


冻结 管理 员 
DD 
en 删除 管理 员 


10.35 管理 员 设置 用 例 图 


10.12.2 ”管理 员 设 置 模块 技术 分 析 


管理 员 可 以 单 击 管理 员 设置 页 面 的 “冻结 ”或 “解冻 ”按钮 ， 对 管理 员 信息 进行 冻结 或 解冻 的 操 
作 。 当 管理 员 登 录 时 ， 首 先 查看 数据 库 中 的 whether 字段 ， 如 果 为 1， 则 说 明 该 用 户 处 于 激活 状态 ， 否 
则 ， 将 显示 提示 窗口 ， 告 诉 该 用 户 ， 此 账号 已 被 冻结 等 提示 信息 。 

对 管理 员 信息 的 冻结 或 解冻 操作 主要 是 根据 传递 的 管理 员 的 id， 将 数据 库 中 的 whether 字段 更 新 。 
程序 代码 如 下 : 

例 程 17 ”代码 位 置 : 光盘 \TM\IO\onlinevadminm freeze_chk.php 

<?php 

session start():; // 开 启 session 支 持 
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include "conn/conn .php": /3 引入 数据 库 链接 文件 
include "inc/chec php": // 引 入 权限 检查 文件 
入 根据 whether 值 ， 进 行 取 反 操作 */ 

if($ POST[whether] — "1") 


S$wt = "0"; 
else 这 $_ POST[whether] 一 "0") 
Swt="1"; 
else{ 
echo "<script>alert(' 非 法 操作 !"):history.go(-1):</script>": 
exit(); 


; 
入 根据 管理 员 id 和 赋值 后 的 Swt， 生 成 update 语 句 */ 
S$o_sqlstr = "update tb manager set whether = ".$wt." where id = ".$ POST[id]: 
谨 执行 update 语 句 */ 
$o_rst = $conn->execute($o_sqlstr): 
这 !(So rst 一 false)){ 
echo "<script>alert( 操 作成 功 ):location='main php?action=manager:</script>": 
?> 
该 方法 实现 很 简单 ， 但 却 十 分 实用 ， 在 实行 会 员 制 的 网 站 上 ， 在 可 以 发 帖 、 回 复 的 论坛 里 ， 都 可 
以 使 用 这 个 方法 来 管理 。 


10.12.3 ”管理 员 添 加 的 实现 过 程 


国 ”管理 员 添 加 使 用 的 数据 表 : tb_manager 

管理 员 可 以 通过 管理 员 设置 页 面 打开 管理 员 信 息 添加 页 面 (addmanagerphp) 。 在 管理 员 信 息 添加 
页 面 中 ,管理 员 可 以 填写 表单 并 将 表单 提交 到 处 理 页 进行 数据 处 理 ， 以 完成 管理 员 信息 添加 的 操作 。 
管理 员 添 加 页 面 的 运行 结果 如 图 10.36 所 示 。 


~ Vp Fe 


管理 员 信 息 洪 加 


管理 员 名 称 : 吏 须 短语 R 
as: 
:ee 
管理 权限 : [视频 并 拓 管 理 员 可 
NRE: ma 

[区 楂 ][ 区 可] 


图 10.36 管理 员 添加 页 面 的 运行 结果 
管理 员 添 加 页 面 涉及 的 HTML 表单 元 素 如 表 10.5 所 示 。 
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表 10.5 管理 员 添加 页 面 涉及 到 的 HTML 表单 元 素 
名 称 类 型 重要 属性 含义 
Tist form method="post" action="addmanager chk.php" 表单 
names text id="names" size="30" 管理 员 名 称 
password password id="password" size="30" 密码 
password2 password id="password2" size="30" 密码 确认 
<option value=" 视 频 目录 管理 员 " selected> 视 频 目录 管理 员 </option> 
<option value=" 音 频 目录 管理 员 "> 音 频 目录 管理 员 </option> 
<option value=" 视 频数 据 管理 员 "> 视 频数 据 管理 员 </option> Bi 
Brade select | option value-" 音 频数 据 管理 员 "> 音频 数据 管理 员 </option> 管理 权限 
<option value=" 会 员 数 据 管理 员 "> 会 员 数 据 管理 员 </option> 
<option value=" 会 员 等 级 管理 员 "> 会 员 等 级 管理 员 </option> 
realname text id="realname" size="30" 真实 姓名 
class="submit" onClick="checkO:" 
Submit2 button Value=" 返 ” 回 " class="submit" onClick="javascript:top.window.closeO" “返回 ”按钮 


管理 员 信息 添加 的 数据 处 理 主要 是 根据 提交 的 表单 数据 ， 在 数据 库 中 添加 对 应 的 记录 ， 首 先 判断 


输入 姓名 是 否 存在 ， 如 果 存 在 , 显示 提示 语 , 并 返回 


添加 新 管理 员 。 程 序 代 码 如 下 : 
例 程 18 代码 位 置 : 光盘 \TM\10\online\admin\addmanager.php 


<?php 
session start(): // 开 启 session 支 持 
include "conn/conn .php": // 载 入 数据 库 链 接 文件 
include "inc/chec.php"; // 载 入 权限 检查 文件 


$a_sql="select * from tb_manager where name=".$_POST[names]."":// 检 查 是 否 有 重 名 
Sa_rst = $conn->execute($a_sql); 

如 果 管 理 员 昵 称 存在 ， 弹 出 提示 框 ， 并 返回 到 上 一 步 。*/ 

这 !Sa rst>EOF) 
echo "<script>alert(' 该 名 称 的 管理 员 已 经 存在 ， 请 更 换 名 称 ');history.go(-1):</script>"; 


else{ 


诬 


让 


诺 


诬 


?> 


诬 


生成 insert 语 句 


$a_sqlstr="insert into tb_manager 
Values(",".$_POST[names].".".$_POST[password].",".$_POST[grade].",".$_POST[realname].”",".date("Y-m-d").",1)"; 


执行 insert 语 句 


eh 


Sa _ Ist] = $conn->execute($a_sqlstr); 
如 果 管 理 员 添 加 成 功 */ 
if(!($a_rst] 一 false)){ 


<script> 


刷新 父 窗口 ， 弹 出 提示 框 ， 最 后 删除 当前 窗口 */ 
top.opener.location.reload(); 
alert(" 管 理 员 添加 成 功 "): 


top.window.close(: 


</script> 


到 上 一 步 。 如果 不 存在 这 种 错误 ,生成 insert 语句 ， 


485 


PHP 项 目 开 发 案例 全 程 实录 (第 2 版 ) 


<?php 
} 


else 
echo "<script>alert(' 添 加 失败 ".$a_sqlstr."):history.go(-1):</script>"; 


10.12.4 ”管理 员 删 除 的 实现 过 程 


国 ”管理 员 删 除 使 用 的 数据 表 : tb_manager 
管理 员 信息 删除 主要 是 根据 传递 的 管理 员 的 id， 将 数据 库 中 对 应 的 数据 删除 。 程 序 代码 如 下 : 
例 程 19 代码 位 置 光盘 \TM\10\online\admin\ del_mfreeze_chk.php 


<?php 
session start(); // 开 启 session 支 持 
include "conn/conn.php"; // 载 入 数据 库 链 接 文 件 
include "inc/chec.php": // 载 入 权限 检查 文件 
S$d_sqlstr = "delete from tb_manager where id= ".$_GET[id]: /生成 delete 语 句 
ifR!($d_rst = $conn->execute($d_sqlstr)) 一 false) // 判 断 删除 是 否 成 功 
echo "<script>alert( 删 除 成 功 "):location=main.php?action=manager';</script>"; 
else 
echo "<script>alert( 删 除 失败 ”):history.go(-1):</script>": 
> 


10.13 ”开发 技巧 与 难点 分 析 


在 本 系统 中 ， 多 个 地 方 使 用 了 无 边框 窗口 和 刷新 父 级 窗口 的 技术 。 下 面 来 学 习 一 下 有 关 这 两 方面 
的 知识 要 点 。 


10.13.1 无 边框 窗口 


平时 看 到 的 浏览 器 窗口 都 是 有 边框 的 ， 如 果 偶 尔 有 一 个 无 边框 的 窗口 弹出 ， 那 么 将 会 更 加 引 人 注 
目 。 实 现 无 边框 窗口 使 用 的 是 JavaScript 脚本 中 的 open 方法 ，open 方法 的 常用 格式 如 下 : 


window.open(url.name,features,replace): 
open 方法 中 的 参数 解释 如 表 10.6 所 示 。 
表 10.6 ”open 方法 的 参数 说 明 


属 性 值 说 明 

ul | 在 弹出 窗口 中 要 打开 的 地 址 

name 要 打开 窗口 的 名 字 。 在 使 用 target 属性 时 会 用 到 ， 可 以 为 空 

features 列举 的 窗口 特征 。 如 果 不 写 任何 参数 ， 那 么 默认 的 参数 多 数 情况 下 是 关闭 的 
Teplace 一 个 boolean 值 ， 指 出 是 否 url 蔡 换 当前 内 容 。 该 参数 可 以 省 略 不 写 
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下 面 应 用 window 对 象 的 open 方法 打开 一 个 指定 大 小 的 窗口 ， 代 码 如 下 : 
<script language="javascript"> 

wopen = open("list.php"."query"."height=500,width=665.scrollbars=no,toolbar=yes,location=yes"): 
</script> 


运行 结果 如 图 10.37 所 示 。 


图 10.37 使 用 open 方法 弹出 窗口 
除了 上 面 实例 中 给 出 的 几 个 窗口 特征 ， 表 10.7 还 列 出 了 其 他 几 个 常用 的 特征 参数 及 说 明 。 
表 10.7 浏览 器 窗口 的 外 观 样式 


新 窗口 的 左 坐标 


新 窗口 的 上 坐标 ， 配 合 lef 可 以 定位 一 个 窗口 的 弹出 位 置 
resizable 是 否 可 以 通过 拖 搜 来 调整 新 窗口 的 大 小 ， 默 认为 no 
在 新 窗口 中 是 否 显示 状态 栏 ， 默 认为 no 


全 注意 : 窗口 特征 是 用 等 号 赋值 ， 喜 号 分 隔 的 ， 在 去 号 或 等 号 前 后 不 要 加 上 空格 ， 因 为 在 有 些 浏览 
器 中 会 显示 错误 。 


10.13.2 ”刷新 父 级 窗口 和 关闭 子 窗口 


在 打开 的 子 窗口 进行 一 系列 的 操作 ， 如 添加 、 删 除数 据 后 ， 就 要 对 当前 窗口 进行 关闭 ， 最 后 返回 
父 级 窗口 中 ， 但 父 级 窗口 中 的 数据 可 能 仍然 是 没有 添加 或 删除 之 前 的 记录 数 ， 这 时 ， 就 可 以 调用 
JavaScript 脚本 , 对 父 窗口 进行 刷新 了 。 这 种 对 父 级 窗口 的 刷新 只 有 对 弹出 窗口 才 有 效 ， 使 用 时 要 注意 。 
该 脚本 的 使 用 格式 为 : 


top.opener.location.reload(): 

opemer 属性 是 子 窗口 的 最 高 层 window 对 象 才 有 的 属性 ,配合 location 对 象 可 以 对 父 窗 口 进行 刷新 
或 重 载 。location 对 象 的 两 个 常用 方法 有 href 方法 和 reload 方法 。href 方法 可 以 刷新 父 窗 口内 容 ， 使 用 
格式 如 下 : 


top.opener.location.href="operation.php": 
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关闭 子 窗口 使 用 如 下 代码 即 可 : 

window.close(): 

不 过 这 里 要 注意 的 是 : close0 方 法 在 关闭 子 窗口 时 , 不 弹出 提示 就 直接 关 掉 浏览 器 了 , 但 对 于 父 级 
窗口 ， 则 弹出 提示 框 ， 经 确认 后 才 可 以 关闭 浏览 器 。 


10.14 ADODB 连接 数据 库 技 术 专 题 


PHP 为 各 种 版 本 的 数据 库 定义 了 庞大 的 函数 群 来 支持 ， 而 且 支 持 得 很 完整 ， 那 为 什么 本 系统 要 使 
用 ADODB 来 对 数据 库 进 行 操作 呢 ? 原因 在 于 ADODB 的 优势 : 一 次 编写 ， 不 需 修改 。 也 就 是 说 ， 当 
系统 使 用 的 数据 库 需 要 更 换 时 ， 只 要 更 新 数据 库 、 数 据 表 即 可 ， 不 需要 再 重新 编写 源码 了 ， 这 为 后 期 
维护 节约 了 一 大 笔 开 销 ， 而 且 不 需要 长 时 间 关 闭 网 站 ， 这 是 PHP 函数 所 无 法 比拟 的 。 本 章 的 技术 专题 
就 来 着 重 学 习 一 下 ADODB 技术 。 


10.14.1 ADODB 简介 


由 于 PHP 存 取 函数 没有 标准 化 , 不 同 数据 库 间 的 函数 名 称 、 参 数 差异 很 大 ， 因 此 在 更 换 数据 库 时 ， 
会 为 用 户 带 来 大 量 的 代码 修复 工作 。 这 时 ， 就 需要 一 组 函数 库 来 隐藏 不 同 资料 库 函数 界面 间 的 差异 ， 
让 开发 者 可 以 很 简单 地 去 切换 资料 库 ， 这 就 是 ADODB。ADODB 目前 支持 MySQL、Oracle、Microsoft 
SQL Server、Sybase、PostgreSQL、Foxpro、Access、ADO 和 ODBC 等 大 多 数 知名 的 、 不 知名 的 数据 
库 类 型 。 


10.14.2 ADODB 的 安装 


要 使 用 ADODB 来 操作 数据 库 ， 首 先 就 要 获取 和 安装 ADODB。 读 者 只 需 到 网 上 下 载 ADODB 包 ， 
解压 到 Web 服务 器 目录 下 即 可 。 


从 注意 : 要 使 用 ADODB， 使 用 的 PHP 必须 是 4.01 以 上 的 版 本 。 
10.14.3 ADODB 常用 函数 及 说 明 


在 10.4.2 节 中 已 经 介绍 过 了 使 用 ADODB 的 一 般 步 骤 ， 在 实际 使 用 中 ， 需 要 根据 情况 进行 不 同 的 
处 理 ， 这 时 就 需要 对 步骤 及 功能 进行 扩展 ， 下 面 就 来 介绍 几 种 经 常会 用 到 的 函数 。 
1. PConnect($host,$user,$password, $database) 


加 ”$host: 数据 库 服 务 器 地 址 。 
回 $user: 数据 库 用 户 名 。 
加 ”S$password: 数据 库 密 码 。 
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加 ”$database: 连接 的 数据 库 。 

如 果 连 接 成 功 回 传 tue， 失 败 则 回 传 false。 

2. SelectLimit($sql,$Snumrows=-1,$offset=-1) 

$sql: 执行 的 sql 语句 。 

$numrows: 显示 的 记录 条 数 。 

S$offset: 从 第 几 条 记录 开始 计算 。 

该 函数 和 MySQL 里 的 limit 子 句 相似 ， 但 要 注意 ，limit 后 面 的 两 个 参数 的 顺序 是 (limit 
offset,numrows) ， 该 函数 则 是 完全 相反 的 。 

3. ErrorMsg() 
回 传 错误 信息 。 当 execute0 函 数 出 现 错误 时 ， 使 用 ErrorMsg0 函 数 可 以 查看 最 后 出 错 的 相关 信息 。 
使 用 方法 如 下 : 


S$recordset = $conn->Execute("select * from tb_audio") 

if(!$recordset) // 判 断 execute0 函 数 执行 中 是 否 有 错误 
echo $conn->ErrorMsg(); 

else{ 


} 

即使 没有 出 现 错误 ，ErrorMsg() 也 会 回 传 最 后 的 状态 值 ， 这 种 情况 下 一 般 不 需要 输出 。 
4. RecordCount() 

回 传 结 果 集 里 的 记录 数 。 使 用 方法 如 下 : 


$sql = 'select * from tb_account'; 
Srs = $conn->Execute($sq]): // 执 行 SQL 语 句 
echo $rs->RecordCount(): // 输 出 查询 结果 数 


5. FieldCount( ) 
回 传 结果 集 里 的 字段 数 。 使 用 方法 同 RecordCount()。 
6. MoveNext( )、Move( )、MoveFirst( )、MoveLast( ) 


加 ”MoveNext(): 移动 指针 到 下 一 条 记录 。 

Move($to): 移动 指针 到 指定 的 列 ($to)。 这 里 使 用 的 是 绝对 定位 ， 如 $to 是 10， 就 移动 到 结果 
集中 的 第 10 条 记录 ， 如 果 $to 大 于 记录 总 数 ， 指 针 将 移动 到 最 后 。 注 意 : 部 分 数据 库 不 支持 
向 后 移动 指针 。 

MoveFirst( ): 实际 上 就 是 Move(0)， 指 针 移动 到 结果 集 头 部 ， 部 分 数据 库 不 支持 该 函数 。 

回 “MoveLast( ): 指针 移动 到 结果 集 的 尾部 ， 也 可 以 表示 为 Move($recordCount() - 1)， 部 分 数据 
库 不 支持 此 函数 。 

最 后 介绍 一 个 ADODB 中 的 常量 。 
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7. $ADODB_FETCH_MODE 

该 常量 决定 了 结果 集 以 哪 种 方式 将 数据 回 传 。 在 有 些 数据 库 中 ， 使 用 $recordset->fields[] 仅 支持 字 
段 编号 为 索引 ， 即 0、1、2 等 ， 不 支持 字段 名 (如 id，user) 作 索 引 ， 这 时 ， 可 以 使 用 该 常量 进行 设置 ， 
该 常量 的 值 有 : 


define( ADODB FETCH DEFAULT'.0): // 系 统 默认 值 

define( ADODB FETCH NUM'.1): /以 字段 编号 为 索引 
define(ADODB FETCH ASSOC'2): // 以 字段 名 称 为 索引 
define( ADODB FETCH BOTH'3): // 同 时 用 时 编号 和 名 称 
这 时 上 面 的 步 又 可 以 改 为 : 


$conn = &ADONewConnection("mysq"); 
$conn->PConnect("db_online"): 

$ADODB FETCH MODE = ADODB FETCH ASSOC: 
Srecordset = $conn->Execute("select * from tb_audio"); 


if(!S$recordset) // 判 断 execute0 函 数 执行 中 是 否 有 错误 
echo $conn->ErrorMsg(); 
else{ 
while(!$recordset->EOF){ // 如 果 没 有 错误 ， 则 配合 wihle 语 句 循环 输出 结果 
echo Srecordset->fields[id]."<br>"; 
Srecordset->movenext(): // 指 针 下 移 


这 里 建议 将 $ADODB_FIELDS_MODED 的 值 固定 为 ADODB_FETCH NUM 及 ADODB FETCH_ 
ASSOC， 不 建议 使 用 ADODB FETCH_BOTH， 因 为 许多 驱动 程序 并 不 支持 该 常量 。 
ADODB 的 常用 函数 就 介绍 这 么 多 了 ， 如 果 读 者 感 兴趣 ， 可 以 下 载 ADODB 手册 来 学 习 更 多 内 容 。 


10.14.4 不 同 数据 库 之 间 的 转换 


使 用 ADODB 的 优势 就 是 可 以 在 以 后 的 升级 中 ， 更 换 不 同 的 数据 库 而 不 用 大 量 地 改动 源 程序 。 但 
不 同 数据 库 之 间 ， 有 的 SQL 语句 或 语法 是 不 相同 的 ， 这 时 就 需要 做 一 些小 幅 的 调整 。 
(1) 在 conn.php 文件 中 ， 更 改 连 接 的 数据 库 类 型 和 参数 。 


Sconn = &ADONewConnection(mysql)): /建立 MySQL 连 接 
Sconn->PConnect("localhost","root"."root","db_online"): // 连 接 “db_online” 数 据 库 


如 使 用 SQL Server 时 ， 上 面 的 连接 语句 更 改 为 : 


$conn = &ADONewConnection(Cmssql): 
$conn->PConnect("localhost"."sa".""."db_online"): 


(2) 注意 包含 时 间 类 型 的 SQL 语句 。 
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不 同 的 数据 库 对 于 时 间 的 表示 法 是 不 同 的 ， 如 MySQL 使 用 时 间 类 型 时 ， 就 像 使 用 字符 类 型 一 样 ， 
放 到 单 引号 里 就 可 以 了 ， 但 Access 则 需要 放 到 “#” 之 间 。ORACLE 虽然 也 是 使 用 单 引 号 ， 但 它 的 格 
式 为 “日 -月 -年 ”， 而 其 他 数据 库 大 多 都 是 “年 -月 -日 ”。 
(3) 尽量 使 用 通用 的 SQL 语句 。 
尽管 国际 标准 化 组 织 (ISO) 在 1987 年 创建 了 SQL 的 一 个 标准 ， 但 在 各 种 SQL 之 间 仍 然 存在 差 
别 。 所 以 为 了 增加 通用 性 ， 最 好 了 解 一 下 数据 库 间 的 不 同 。 例 如 ，MySQL 中 的 limit 子 名 在 oracle 中 
就 不 被 支持 。 


10.15 本 章 总 结 


online 影视 365 网 实现 了 一 个 在 线 播放 平台 。 该 网 站 具备 一 个 影视 网 站 所 需要 的 所 有 的 基本 功能 ， 
包括 影视 文件 的 上 传 、 下 载 、 播 放 及 介绍 。 还 介绍 了 js 脚本 语言 的 一 些 实用 技巧 ， 最 后 ， 介 绍 了 本 章 
的 核心 内 容 : ADODB 连接 数据 库 技术 。 希 望 通过 本 章 的 学 习 ， 读 者 不 仅 可 以 建立 一 个 自己 的 影视 网 ， 
而 且 可 以 熟练 使 用 ADODB 来 连接 任意 一 个 数据 库 。 有 条 件 的 读者 可 以 试 着 将 本 系统 的 数据 库 改 为 
SQL Server 或 ORACLE， 来 增加 对 ADODB 的 了 解 及 熟练 程度 。 


NA 


在 全 球 信息 化 高 速 发 展 的 今天 ， 互 联网 已 经 成 为 资源 共享 、 互 动 、 交 流 、 办 公 
及 娱乐 的 最 为 广泛 和 有 效 的 平台 ， 如 何 高 效 地 利用 这 些 资源 已 经 成 为 网 民 们 最 为 关 
注 的 问题 。 百 度 、 谷 歌 等 互联 网 搜索 工具 的 出 现在 很 大 程度 上 提高 了 信息 共享 的 进 
程 。 本 章 所 介绍 的 “明日 知道 ”网 站 是 针对 IT 技术 问题 进行 搜索 ， 为 程序 开发 人 
员 提 供 交 互 的 平台 ， 与 互联 网 上 一 些 知 名 的 搜索 工具 相 比 更 具 专 业 化 的 特点 。 
通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


明日 知道 网 站 


(Zend Framework 实现 ) 
(器 视频 讲解 : 1 小 时 20 分 钟 ) 


网 站 项 目 开 发 的 基本 流程 

Zend Framework 框架 的 搭建 方法 
MVC 开发 模式 的 应 用 

类 似 Google 搜索 条 的 制作 方法 
查询 结果 的 关键 宁 描 红 输 出 

站 在线 编 辑 路 的 原理 及 制作 方法 


豆 吾 吾 至 
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11.1 开发 背景 


XX 科技 有 限 公 司 是 一 家 以 IT 教育、 互联 网 开发 和 运营 为 主 的 高 科技 互联 网 企业 。 在 多 年 的 经 营 
中 积累 了 大 量 的 IT 技术 方面 的 资源 ， 为 了 提升 公司 的 知名 度 ， 使 资源 得 以 共享 ， 并 创造 更 多 的 广告 利 
润 ， 公 司 组 织 了 一 支 团 队 来 开发 “明日 知道 ”网 站 ,争取 打造 一 个 国内 知名 的 专业 IT 技术 搜索 和 交互 
平台 。 

“明日 知道 ”网 站 是 为 了 使 搜索 变 得 更 加 专业 ， 以 方便 广大 程序 相关 工作 者 而 打造 的 专业 化 互联 
网 搜索 引擎 。 通 过 该 搜索 引擎 ,程序 开发 人 员 可 以 方便 地 查找 到 有 关 软 件 开发 方面 的 技术 问题 和 文章 。 
并 且 “ 明 日 知道 ”网 站 提供 了 互动 平台 ， 用 户 在 学 习 编程 过 程 中 如 果 遇 到 了 技术 问题 ， 可 以 将 问题 发 
表 于 明日 知道 网 站 上 ， 其 他 注册 会 员 就 可 以 对 提出 的 问题 进行 解答 ， 这 样 即便 通过 “明日 知道 ”没有 
搜索 到 要 查找 的 内 容 ， 也 可 以 通过 互动 方式 将 问题 解决 。 专 业 化 搜索 工具 已 经 成 为 网 上 资源 相当 丰富 
下 的 新 生 儿 ， 是 具有 一 定 发 展 潜能 的 项 目 课题 。 


11.2 需求 分 析 


对 于 “明日 知道 ”之 类 的 信息 检索 网 站 来 说 ， 具 备 功能 强大 的 搜索 模块 显得 尤为 重要 ， 借 鉴 其 他 
知名 的 搜索 工具 ， 方 便 、 高 效 的 搜索 引擎 应 该 具备 关键 字 提 示 列 表 、 分 词 查 询 和 查询 结果 的 描 红 输 出 
等 功能 。 为 了 避免 用 户 要 查找 的 内 容 在 本 站 没有 查找 到 结果 ， 所 以 还 需要 增加 用 户 交 流 模块 ， 这 样 即 
使 用 户 的 问题 在 本 站 没有 查找 到 答案 ， 也 可 以 将 问题 发 布 出 去 ， 其 他 用 户 就 可 以 对 该 问题 进行 回答 
直到 解决 为 止 。 所 以 “明日 知道 ”网 站 应 该 打造 一 个 服务 开发 人 员 ， 对 资源 进行 不 断 积累 的 综合 平台 。 

通过 实际 调查 ， 要 求 “ 明 日 知道 ”网 站 应 该 具备 以 下 功能 : 
类 似 百度 和 Google 等 搜索 引擎 的 搜索 条 。 

多 关键 字 的 分 词 查询 。 

查询 结果 的 关键 字 描 红 输出 。 
使 用 在 线 编辑 器 发 布 和 回复 问题 。 
问题 的 引用 。 


回回 罗 加 加 


11.3 功能 结构 


“ 磨 刀 不 误 砍 柴 工 ”， 建 立 一 个 完善 的 系统 功能 图 对 项 目的 整体 开发 具有 指导 意义 ， 因 此 在 项 目 
编码 前 在 此 给 出 项 目前 台 和 后 台 的 系统 功能 结构 图 。 其 中 “明日 知道 ”网 站 的 前 台 功 能 结构 图 如 图 11.1 
所 示 。 

“明日 知道 ”网 站 的 后 台 功 能 结构 图 如 图 11.2 所 示 。 
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柄 豆 痊 回 


图 11.1 网 站 前 台 功 能 结构 图 图 11.2 网 站 后 台 功 能 结构 图 
11.4 功能 预览 


“明日 知道 ”网 站 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 
几 个 典型 功能 的 页 面 ， 其 他 页 面 可 参见 光盘 中 的 源 程序 。 

“明日 知道 ”网 站 的 首页 由 网 站 Logo、 语 言 导航 条 、 搜 索 区 、 友 情 链接 和 版 权 声明 等 部 分 组 成 。 
页 面 简洁 大 方 、 突 出 核心 ， 便 于 用 户 使 用 ， 其 运行 效果 如 图 11.3 所 示 。 


明日 MING RI Ea 


evs 护 程 河 由 java web jeva jsvs 从 入 [ 
明日 图 书 | 明日 软件 | 编程 间 典 
扶 术 服务 执 晓 : 400.675-1066 仿真 : 0431.34939777 全 业 闻 入 : mingizoft@mingrieoft com 


全 司 地 址 ; 言 林 省 长春 市 二 道 区 夺 归 大 街 39 号 亚 泰 广场 C 座 ” 言 JCP 备 07500273 导 
Copyrgh! © ww mingrihook com Al Rights Reservedl 


图 11.3 明日 知道 首页 


会 员 登 录 模 块 完 成 注册 会 员 的 登录 操作 ， 并 且 融 入 验证 码 技术 ， 同 时 加 入 了 会 员 注册 的 超 链接 ， 
其 运行 效果 如 图 11.4 所 示 。 

问题 分 类 模块 展示 “明日 知道 ”网 站 中 专题 技术 的 分 类 ， 同 时 展示 每 类 中 包含 的 问题 和 回复 情况 ， 
其 运行 效果 如 图 11.5 所 示 。 

输出 查询 结果 模块 展示 搜索 的 结果 ， 并 且 对 查询 的 关键 字 进 行 描 红 ， 同 时 对 查询 结果 进行 分 页 输 
出 ， 其 运行 效果 如 图 11.6 所 示 。 


ep 
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四 是 人 年关 工作 多 避 序 员 ， 对 一 此 代码 返还 不 是 候 朱 ,对 代 史 的 书写 下 是 不 恨 开 范 。 公 司 要 交 开 发 一 个 项 目 ， 匠 有 二 各 不 看 头 半 ,朋友 交响 村 司 我 看 一下， 了 福 很 用 ， 里 硬 的 
长 玛 是 搞 扣 好 的 ， 直接 了 可 以 实用 ,省 了 我 不 少 的 时 间 利 ， 经 更 涪 和 的 快 ， 冶 和 发 了 奖 全 ， 有 机 一定 要 请 你 fp2th 呢 
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图 11.6 查询 结果 的 分 页 输出 页 面 
11.5 数据 库 设 计 


在 进行 项 目 开发 前 需要 根据 实体 属性 建立 数据 库 实体 模型 ， 然 后 根据 实体 之 间 存在 的 客观 联系 设 
计 出 合理 的 数据 表 。 由 于 数据 库 是 项 目 内 容 的 主要 载体 ， 所 以 设计 出 合理 的 数据 库 表 及 表 间 关系 显得 
非常 重要 。 


11.5.1 数据 库 分 析 


“明日 知道 ”网 站 采用 PHP 语言 开发 ， 并 且 该 项 目 为 资源 搜索 型 网 站 ， 数 据 量 会 很 大 ， 所 以 稳定 
的 数据 库 、 高 效 的 存储 和 检索 能 力 显得 非常 重要 。 综 合 上 述 因素 及 PHP 的 完全 跨 平 台 等 特性 ， 应 该 首 
选 MySQL 数据 库 作 为 “明日 知道 ”网 站 数据 的 载体 。 开 发 人 员 在 长 时 间 的 开发 工作 中 也 总 结 出 ，PHP 
和 MySQL 是 最 佳 的 黄金 搭档 。 
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11.5.2 ”数据 库 概念 设计 


根据 “明日 知道 ”网 站 的 需求 和 功能 结构 分 析 ， 规 划 出 整个 项 目的 数据 库 实体 应 该 包括 问题 类 别 
实体 、 关 键 字 实体 、 问 题 实体 、 回 复 实体 和 用 户 实体 等 部 分 。 下 面 介 绍 这 几 个 实体 的 E-R 图 。 

1， 问 题 类 别 实体 

“明日 知道 ”网 站 定位 打造 一 个 专业 、 高 效 的 在 线 搜索 工具 ， 这 样 就 需要 将 搜索 内 容 按 类 别 划分 。 
资源 分 类 明确 ， 可 以 有 效 地 提高 用 户 检索 的 效率 ， 方 便 用 户 定位 查找 。 问 题 类 别 实体 包括 类 别 ID 号 、 
类 别名 称 、 类 别 描述 和 类 别 添加 时 间 等 属性 ， 其 实体 E-R 图 如 图 11.7 所 示 。 


添加 时 间 


图 11.7 问题 类 别 实体 E-R 图 


2. 问题 实体 

问题 实体 是 对 用 户 提 出 问题 的 抽象 ， 根 据 功 能 分 析 ， 问 题 实体 包括 问题 ID 号、 主题 、 内 容 、 添 加 
时 间 、 置 顶 标识 、 提 问 用 户 ID 号、 问题 类 别 ID 号 、 浏 览 次 数 、 上 传 文件 名 和 精华 问题 标识 等 属性 ， 
其 实体 E-R 图 如 图 11.8 所 示 。 


精华 问题 标识 


图 11.8 问题 实体 E-R 


3. 回复 实体 
回复 实体 包括 回复 ID 号 、 主 题 、 内 容 、 最 佳 答案 标识 、 添 加 时 间 、 用 户 ID、 回 复 问题 ID、 问 题 
类 别 ID 等 属性 ， 其 实体 E-R 图 如 图 11.9 所 示 。 

4. 关键 字 实 体 

关键 字 实体 是 对 用 户 输入 关键 字 进 行 记录 的 抽象 ， 该 实体 包括 关键 字 名 称 和 关键 字 被 检索 次 数 等 
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属性 ， 其 实体 E-R 图 如 图 11.10 所 示 。 


图 11.9 回复 实体 E-R 图 


图 11.10 ”关键 字 实体 E-R 图 


5. 用 户 实体 

用 户 实体 是 对 网 站 注册 用 户 的 抽象 ， 其 属性 包括 用 户 ID、 用 户 昵称 、 登 录 密码 、E-mail、 联 系 电 
话 、 所 在 地 、 注 册 时 间 、 头 像 名 称 、 发 帖 次 数 、 回 帖 次 数 、 积 分 、 用 户 类 别 标识 等 属性 ， 其 实体 E-R 
图 如 图 11.11 所 示 。 


图 11.11 用 户 实体 E-R 
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创建 数据 库 及 数据 表 


根据 需求 分 析 及 数据 库 概 念 设计 ， 可 以 制定 出 “明日 知道 ”网 站 数据 库 (这 里 命名 为 db_known) 
应 该 包括 问题 类 别 表 (tb_bbstype)、 问 题 表 (tb_title)、 问题 回 复 表 (tb_reply)、 关 键 字 表 (tb_keywords) 
和 用 户 表 (tb_user) 等 5 个 表 ， 其 中 各 表 的 主要 参数 如 图 11.12 所 示 。 

下 面 给 出 “明日 知道 ”数据 库 中 各 个 数据 表 字 段 的 具体 描述 。 


1. 问题 类 别 表 (tb_bbstype) 


问题 类 别 表 用 于 存储 问题 类 别 ， 其 结构 如 图 11.13 所 示 。 


表 类 型 整理 大 小 
tw_ pbstype MySAM utr_unicode_ci 26KB 字 息 类 型 整理 尾 性 NAwW 默认 额外 
th_keywords MyISAM utfg_unicode_cl 45KB 到 int(11) 否 auto_increment 
th_reply MyISAM utfB_unicode_ci 47KB typename varchar(50) utre_unicode_ci 理 
tb_tive MyISAM 。 ut8_unicode_cl 66KB addtme datetime 否 
WD_user MylSAM utfe_unicode_ci 108KB yarchar(255) utf8_unicode_ci 否 
11.12 “明日 知道 ”网 站 数据 表 主 要 参数 图 11.13 ”问题 类 别 表 结 构 


2. 问题 表 (tb_title) 
问题 表 用 于 存储 问题 内 容 ， 其 结构 如 图 11.14 所 示 。 


int(11) 

varchar(200) ut8_unicode_ci 
tod ut8_unicode_ci 
datetime 

tinyint(1) 

mt) 

int(11) 

Intt1) 


auto_increment 


varchar(50) utf9_unicode_ci 
tinyint(1) 
unpbnicontent text ut9_unicode_cl 


串 商 各 和 | 放 如 间 翅 巷 


图 11.14 问题 表 结 构 


3， 问题 回复 表 (tb_reply) 


问题 回 


复 表 用 于 存储 用 户 对 问题 的 回复 ， 其 结构 如 图 11.15 所 示 。 


类 性 NM 默认 额外 
到 int(11) auto_increment 
ye varchar(200) utB_unicode_ci NULL 
content texd ut8_unicode _eci NULL 
addime datetime NULL 
userid int(11) NULL 
Weid int(11) NULL 
topindex int(11) NULL 


bbstype_id int(11) NULL 


图 11.15 问题 回复 表 结 构 
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4. 关键 字 表 (tb_keywords) 
关键 字 表 用 于 存储 用 户 曾 搜索 的 关键 字 及 每 个 关键 字 被 搜索 的 次 数 ， 其 结构 图 如 图 11.16 所 示 。 


字 和 让 类 型 整理 尾 性 Nwi 歌 认 般 外 
型 int(11) 再 auto_increrent 
keyword Yarchar(50) utfe_unicode_ci 香 
searchtime int(11) 否 


图 11.16 关键 字 表 结构 


5. 用 户 表 (tb_user) 
用 户 表 用 于 存储 用 户 信息 ， 其 结构 图 如 图 11.17 所 示 。 


默认 额外 
auto_increment 


过 int(11) 

password varchar(50) utrB_unicode_ci 
nemname varchar(50) utre_unicode_ci 
email Varchar(100) utf8_unicode_ci 
tef varchar(20) ut8_unicode_cl 
pc varchar(20) ut8_unicode_ci 
regtme datetime 
和 ce 
pubtimes 
replytimes 
usertype 


varchar(50) utf8_unicode_ci 
int(11) 
int(11) 
int(11) 


各 各 各 沽 各 各 各 沽 加 站 叫 芝 
峡 
六 


图 11.17 用 户 表 结构 
11.6 技术 攻关 


“明日 知道 ”网 站 在 开发 过 程 中 ， 多 处 使 用 了 较 新 的 技术 。 为 了 便于 对 该 项 目 各 个 模块 的 理解 ， 
本 节 首 先 对 “明日 知道 ”网 站 所 使 用 的 关键 技术 进行 讲解 。 


11.6.1 类似 Google 搜索 引擎 的 搜索 条 


为 了 方便 网 站 浏览 者 检索 网 站 内 容 ， 很 多 知名 网 站 的 查询 关键 字 文 本 框 都 提供 了 关键 字 提 示 下 拉 
列表 框 ， 这 样 用 户 就 可 以 根据 以 往 所 输入 的 关键 字 更 加 准确 地 定制 自己 的 关键 字 ， 从 而 在 很 大 程度 提 
高 了 要 查询 内 容 被 高 效 、 准 确 查 询 出 来 的 几率 。 同 样 ， 制 作 “ 明 日 知道 ”网 站 的 搜索 模块 也 引用 了 有 
关键 字 提 示 的 搜索 条 ， 如 图 11.18 所 示 。 

那么 , 类 似 Google 的 搜索 条 是 如 何 实现 的 呢 ? 是 不 是 特别 复杂 呢 ? 继续 学 习 下 面 内容 将 会 寻求 到 
答案 。 在 具体 讲解 实现 过 程 前 ， 首 先 讲解 制作 该 搜索 条 的 思路 ， 当 用 户 在 关键 字 文 本 框 中 输入 内 容 时 ， 
通过 文本 框 的 onkeyup 事件 调用 用 来 显示 关键 字 下 拉 列 表 的 方法 ， 该 方法 通过 Ajax 技术 〈 这 里 使 用 
Ajax 的 框架 jQuery)， 应 用 GET 方法 向 服务 器 发 送 请 求 ， 请 求 过 程 中 将 用 户 输入 的 关键 字 发 送 给 服务 
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器 ， 服 务 器 经 过 处 理 返 回 所 有 以 该 关键 字 开头 的 用 户 曾 搜索 的 关键 字 及 搜索 次 数 和 与 该 关键 字 匹 配 的 
结果 个 数 。 下 面具 体 讲解 类 似 Google 搜索 条 的 实现 方法 。 


图 11.18 类似 Google 搜索 引擎 的 搜索 条 


(1) 建立 用 于 录入 关键 字 的 文本 框 ， 这 里 使 用 Zend Framework 框架 的 formTextO 视 图 助手 建立 。 
Zend Framework 框架 的 视图 助手 ， 大 部 分 用 来 生成 组 件 和 实现 自动 转 义 变量 的 功能 。 另 外 ， 有 些 助 手 
用 来 创建 基于 路 由 的 URL 和 HTML 列表 以 及 声明 变量 。 这 里 所 使 用 的 formText0 助 手 用 来 生成 HTML 
的 文本 框 组 件 。 
formText() 视 图 助手 的 语法 格式 如 下 : 


formText(string name, string value, array attribs) 
参数 说 明 如 表 11.1 所 示 。 
表 11.1 formText() 视 图 助手 的 参数 说 明 


参 数 说 了 明 

name 必要 参数 ， 指 定 所 生成 文本 框 组 件 的 名 称 

value 可 选 参数 ， 文 本 框 的 内 容 ， 默 认为 null 

attribs 可 选 参数 ， 以 数组 形式 指定 文本 框 的 多 个 属性 ， 默 认为 null 


“明日 知道 ”网 站 使 用 formTextO 视 图 助手 生成 搜索 文本 框 的 代码 如 下 : 

例 程 01 代码 位 置 : 光盘 \TM\11\known\application\modules\default\views\scripts\index\index.phtml 

echo S$this->formText(keywords'. ". arayf(style 一 >'width:3S0px: height:26px: line-height:26px: font-size:15px: 
Color:#333333; border:0px:"，  'onkeyup'=>'showKeywordsList)'. ‘onkeydown'=>'selectListO', ‘autocomplete'=>'off)) 
/通过 formTextO 视 图 助手 输出 查询 关键 字 录 入 文本 框 

上 述 代 码 中 ， 将 关键 字 文 本 框 命名 为 keywords， 默 认 值 为 空 串 。 在 属性 数组 中 ， 通 过 键 名 为 style 
的 数组 元 素 指定 文本 框 样式 ， 通 过 键 名 为 onkeyup 的 数组 元 素 指 定 当 按键 抬 起 时 所 触发 的 事件 ， 通 过 
键 名 为 onkeydown 的 数组 元 素 指 定 当 按键 按 下 时 所 触发 的 事件 ， 最 后 为 键 名 为 autocomplete 的 数组 元 
素 指 定 off 值 来 去 除 文本 框 自身 的 下 拉 列 表 提 示 。 

(2) 当 用 户 在 文本 框 中 录入 关键 字 时 ， 将 调用 自 定义 的 showKeywordsList0 方 法 显示 关键 字 下 拉 

列表 ， 代 码 如 下 : 
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例 程 02 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\views\scripts\index\index.phtml 


var index = 0: // 关 键 字 列表 项 索引 
var count = 0: // 关 键 字 列表 总 数 
function showKeywordsListO{ // 显 示 关键 字 下 拉 列 表 
Var keycode = eventkeyCode: // 用 户 按 键 的 ASCIH 码 
if(keycode != 40 && keycode != 38){ // 如 果 不 是 向 上 和 向 下 方向 键 
index= 0: /关键 字 索 引 归 0 
if($("#keywords").val) — "" | $.trim($("#keywords").val)) — ""){ 
S$("#searchLayer").css("display", "none"): /如 果 用 户 输入 空 字符 或 空格 ， 则 隐藏 下 拉 列 表 
jelse{f 
© setTimeout("reSearch|)", 150); // 否 则 每 隔 150 毫 秒 调用 一 次 reSearch0 方 法 
} 
} 
< 代码 贴 二 


@ 这 里 之 所 以 要 间隔 150 毫秒 后 调用 reSearch0 方 法 ， 是 为 了 降低 连续 对 服务 器 请 求 而 带 来 的 负载 。 


上 述 代 码 中 ， 首 先 使 用 Javascript 的 event 对 象 的 keyCode 属性 获取 用 户 输入 字符 的 ASCII 码 ， 然 
后 判断 该 ASCII 码 是 否 为 40〈 代 表 向 上 方向 键 ) 或 38 (代表 向 下 方向 键 )， 如 果 不 是 这 两 个 方向 键 ， 
并 且 用 户 录入 的 字符 不 为 空 串 或 空格 ， 则 间隔 150 毫秒 调用 reSearch0 方 法 来 显示 与 用 户 录入 关键 字 所 
匹配 的 关键 字 搜索 记录 。 其 中 reSearch() 方 法 的 代码 如 下 : 

例 程 03 ”代码 位 置 ; 光盘 \TM\11\known\application\modules\default\views\scripts\index\index.phtml 


function reSearchO{ 
if($("#keywords").val) != ""){ /如 果 用 户 录入 的 关键 字 不 为 空 
@ $.get("<?php echo $this->baseUrl(Vindex/keywords-lisUkeyword/"+encodeURI($("#keywords").valO)+")2>"， 
null, function(data){ 
// 通 过 jQuery 向 服务 器 发 送 GET 请 求 
if($.trim(data) — ""){ // 如 果 返 回 数据 为 空 ， 则 隐藏 下 拉 列 表层 
S$("#searchLayer").css("display", "none"):; 
}else{ 
S("#searchLayer").html(data): // 如 果 返 回 数据 不 为 空 ， 则 显示 下 拉 列 表 
S$("#searchLayer").css("display", "block"): 
@ count = parseInt($("#totalList").val0): /获取 列表 中 关键 字 个 数 


< 人 代码 贴 二 
@ encodeURIO 方 法 : 用 来 对 关键 字 进 行 URL 编码 。 
@ parseInt( 方 法 : 用 来 将 字 串 转换 为 整 型 。 
上 述 代码 中 ， 通 过 jQuery 的 $.get( 方 法 向 服务 器 发 送 请 求 ， 如 果 返 回 结果 不 为 空 素 ， 则 将 内 容 显 示 在 下 拉 列 表 中 。 


[DD 说 明 : 通过 Ajax 技术 向 服务 器 发 送 请 求 ， 传 递 的 内 容 应 该 为 UTF-8 编码 ， 如 果 不 是 ， 应 该 使 用 
函数 iconvO 对 传递 的 内 容 进 行 转 码 。。 
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(3) 当 用 户 移动 键盘 的 上 下 方向 键 时 ， 列 表 中 所 选中 的 项 以 特殊 色 显示 ， 并 在 关键 字 文本 框 中 显 
示 当 前 选中 项 的 内 容 ， 代 码 如 下 : 
例 程 04 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\views\scripts\index\index.phtml 


function selectListO{ 
Var keycode = eventkeyCode: // 获 取 用 户 录入 字符 的 ASCII 
if(keycode — 40){ // 如 果 用 户 按 向 下 方向 键 
$("#istItem "+index).css("background-color", "#158CD0"): 。“ // 更 改 当 前 所 选项 的 背景 色 
S$("#listItem "+index).css("color", "#FFFFFF"): // 更 改 当 前 所 选项 的 前 景色 
$("#keywords").val($.trim($("#listItem li "+index).html0)): ” // 将 当前 所 选项 显示 在 文本 框 中 
iflindex > O){ 


S$("#listItem "+parseInt(index-1)).css("backeround-color", "#FFFFFF"); 
// 更 改 当 前 项 前 一 项 的 背景 色 
$("#listItem "+parseInt(index-1)).css("color", "#333333"); ”// 更 改 当 前 项 前 一 项 的 前 景色 


} 
这 index < count-D){ // 如 果 当 前 索引 小 于 总 关键 字数 减 1 
index++: /索引 数 增 1 
} 
}else if(keycode 一 38){ // 如 果 用 户 按 向 上 方向 键 
这 index> O){ /如果 索 引 大 于 0 


S$("#listItem "+parseInt(index-1)).css("background-color", "#158CDO"): 
// 更 改 当前 项 前 一 项 背景 色 
$("#listItem_"+parseInt(index-1)).css("color", "#FFFFFF"); // 更 改 当 前 项 钱 一 项 前 景色 
S$("#keywords").val($.trim($("#listItem li_"+parseInt(index-1)).htmlO0)): 
// 将 当前 所 选项 显示 在 文本 框 中 
} 
S$("#listItem "+index).css("backeground-color", "#FFFFFF"): 
// 更 改 当前 项 背景 色 


S$("#listItem "+index).css("color", "#333333"):; // 更 改 当 前 项 前 景色 
iflindex > D){ /如 果 索 引 大 于 1 
index--: /索引 减 1 


} 
} 
} 


上 述 代码 首先 获取 用 户 键入 字符 的 ASCI 码 ， 然 后 判断 该 值 是 否 为 向 上 方向 键 或 向 下 方向 键 所 对 
应 的 ASCI 码 ， 如 果 是 则 通过 jQuery 的 ss( 方 法 改变 所 选项 的 前 景色 和 背景 色 ， 从 而 达到 关键 字 列 表 
中 内 容 上 下 切换 的 效果 。 

(4) 在 上 文 的 讲解 中 ， 已 经 提 到 客户 端 应 用 GET 方法 向 服务 器 发 出 请 求 ， 那 么 发 出 请 求 后 做 了 
哪些 处 理 呢 ? 从 下 述 代码 中 将 会 找到 答案 。 

例 程 05 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\controllers\IndexController.php 

public function keywordsListAction () 


by header('content-type:text/html: charset=utf-8"): // 设 置 页面 编 码 为 utf-8 

Skeyword = urldecode($this-> request->getParam('keyword')): // 获 取 用 户 录入 的 关键 字 并 进行 URL 解 码 
@ Skeywordses = $this-> keywordsModel->findByLike($Skeyword. 10. te): ”// 调 用 关键 字模 型 的 findByLink0 方 
法 查询 所 有 以 该 关键 字 开 头 的 内 容 


S02 
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Stotal = 0: /保存 总 匹配 关键 字数 
if ($keywordses != null) { 
Stotal = count($keywordses): // 统 计 总 匹配 关键 字数 
b 
S$str = "<input type="hidden" name="totalList" id="totalList" value="" . $total . " />"; 
foreach ($keywordses as $k => Skey) { // 通 过 循环 形成 下 拉 列 表 
S$str = '<div id="listItem ' . $k . " style="width:98%:; clearboth: background-color:#FFFFFF; cursor:pointer;" 
‘onmouseover="this.style.backeroundColor=\#158CDO\'; this.style.color=\#FFFFFF\" 


‘onmouseout="this.style.backeroundColor=\#FFFFFF\; this.style.color=\'#333333\" onclick="setText(\" . $key['keyword'] . 
WW) 


$str .= "<ul>'; 
Sstr = <li id="listItem li' . $k . " style="width:60%:; height:18px: line-height:18px: text-align:left: 
float:left:">' . $key[keyword] . '</li>"; 
$str = " <li style="width:38%; height:18px: line-height:18px; text-align:right: float:right;">' . 
$key['searchtime'] . "</li>": 
S$str ='</ul>': 
Sstr = "</div>"; 
| 
echo $str; 
Sthis->_helper->layout->disableLayout(): // 取 消 页 面 布局 
Sthis->_helper->viewRenderer->setNoRender(): /取消 视 
， 
< 代码 贴 十 


@ header0O 函 数 用 于 向 客户 端 浏览 器 发 送 HTTP 头 信息 ， 这 里 用 来 指定 页 面 所 使 用 的 编码 为 UTF-8。 
@ 这 里 调用 关键 字模 型 的 findByLike0 〇 方法， 获取 关 键 字 被 搜索 最 多 的 10 条 记录 。 


上 述 代 码 为 default 模块 中 IndexController 控制 器 的 keywordsListAction 动作 ， 主 要 用 来 获取 下 拉 
列表 的 数据 并 发 送 给 客户 端 。 

由 于 上 述 代码 只 需要 显示 下 拉 列 表 内 容 ， 不 需要 显示 网 站 头 和 尾 信息 ， 即 不 需要 使 用 布局 ， 所 以 
使 用 如 下 代码 取消 该 动作 的 页 面 布 局 。 


S$this->_ helper->layout->disableLayoutO: 
全 注意; 使 用 headerO 函 数 发 送 HTTP 头 信息 应 该 在 所 有 页 面 输出 之 前 进行 。 


在 Zend Framework 中 , 使 用 当前 控制 器 对 象 的 _ help 属性 的 layout 属性 的 disableLayout0 方 法 取消 
布局 。 

同样 ， 由 于 不 需要 视图 ,直接 通过 echo 语句 就 可 以 输出 列表 中 的 内 容 ， 所 以 可 以 将 该 Action 所 使 
用 的 视图 通过 如 下 代码 取消 。 

S$this->_helper->viewRenderer->setNoRender(): 


从 注意 : 由 于 “明日 知道 ”网 站 使 用 Zend Framework 框架 开发 ， 因 此 学 习 本 章 内 容 需 要 读者 具有 一 
定 MVC 开发 思想 和 Zend Framework 框架 基础 。 
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11.6.2 ”使 用 空格 分 隔 多 关键 字 


用 户 在 使 用 百度 、Google 等 互联 网 搜索 工具 时 ， 如 果 对 多 个 关键 字 同 时 进行 搜索 ， 可 以 将 多 个 关 
键 字 用 空格 分 隔 ， 从 而 提高 查找 效率 。 

“明日 知道 ”网 站 中 为 了 方便 用 户 查找 ， 采 用 分 词 技术 可 以 利用 多 个 关键 字 进 行 同 时 查找 。 例 如 
用 户 输入 “编程 ”和 “软件 ”两 个 关键 字 ， 并 且 关键 字 用 空格 分 隔 ， 提 交 搜 索 后 将 查找 出 所 有 包含 这 
两 个 关键 字 的 内 容 ， 如 图 11.19 所 示 。 


我 是 今年 刚 工作 的 程序 员 ， 码 的 运用 还 不 是 很 熟悉 ， 对 代码 的 书写 还 是 不 很 规范 。 公 司 要 我 开发 一 个 项 目 ， 我 有 点 损 不 着 头 
绪 ， 朋 友 买 的 编程 词典 我 看 就 得 很 实用 ， 里 面 的 代码 是 编辑 好 的 ， 直 接 就 可 以 实用 ， 省 了 我 不 少 的 时 间 啊 ， 经 理 还 说 我 做 的 
快 ， 给 我 发 了 奖金 , 有 机 会 你 们 吃饭 呢 

和 2 ”10-03-31 02;45 ”明日 知 乔 


查 
的 易学 、 易 查 、 易 用 的 超 媒 体 编 程 学 习 软 件 实例 、 方 案 项目、 视频 、 界 面 、 工 具 等 
数 十 万 开发 资源 。 通 过 编程 辣 典 ， 学 习 者 可 以 快速 学 习 编程 知识 ， 快 速 学 握 编 程 法 ,并 在 编程 中 体会 快乐 和 乐趣 ,培养 信 心 ， 


迅速 成 为 编程 高 手 。 对 于 开发 者 ， 不 但 可 以 高 效 查询 相关 技术 ,快速 借用 现 有 实例 、 项 目 、 方 案 、 源 码 快速 搭 
Wwwmingrizd com/question/thread-1 10.03.31 0236 ”明日 知 这 


图 11.19 多 关键 字 同 时 查询 
下 面 来 一 起 探讨 这 种 实用 的 分 词 技术 的 实现 方法 。 
(1) 当 用 户 输入 完 关 键 字 提交 查询 内 容 后 ， 首 先 在 InNdexController 控制 器 的 searchAction 动作 方 
法 中 使 用 getParam() 方 法 获取 用 户 提交 关键 字 的 值 ， 代 码 如 下 : 
$keywordsStr = urldecode($this->_request->getParam('keywords")): 
Zend Framework 框架 中 接收 页 面 间 传 递 参数 的 方法 有 3 种 ， 分 别 为 当前 控制 器 对 象 _request 属性 
的 getQuery0 方 法 、getPost0 方 法 和 getParam() 方 法 。 这 3 种 方法 的 区 别 是 : getQuery0 方 法 用 来 获取 
GET 方法 提交 的 数据 ，getPost0 方 法 是 用 来 获取 POST 方法 提交 的 数据 ;而 通过 getParam() 方 法 既 可 以 
获取 GET 方法 提交 的 数据 ， 还 可 以 获取 POST 方法 提交 的 数据 。 
(2) 在 主题 表 (tb_title) 模型 Model DbTable_Title 中 建立 findByLike0 方 法 ， 用 来 对 用 户 输入 的 
关键 字 进 行 模糊 查找 ， 代 码 如 下 : 
例 程 06 ”代码 位 置 ， 光盘 \TM\11\known\application\models\DbTable\Title.php 
public function findByLike ($keywordsStr, $bbstype_id = 0, SorderFlag = null, Spage = 1. SpageSize = 10. SpageRange = 


5) 
{ 
© SarrayKey Words = explode(' ', $keywordsStr): /将 多 个 关键 字 用 空格 分 隔 保存 到 数组 中 
if($orderFlag 一 null || SorderFlag — 0) { // 指 定 排序 方式 
S$order = 'addtime desc'; // 时 间 降 序 
}else{ 
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S$order = 'browse desc': // 时 间 升 序 

b 

Sselect = $this->getAdapter0->select0: // 生 成 Zend_Select 对 象 

Sselect->from($this-> name): // 为 Zend_Select 对 象 指定 表 名 

foreach ($arrayKeyWords as $keywords) { /遍历 所 有 被 空格 分 隔 的 关键 字 
让 (trim($keywords) (=") { /如果 关键 字 不 为 空 串 

© $select->orWhere($this->getAdapter0->quoteIntof'title like ?','%'. Skeywords .'%")): 
// 对 主题 进行 匹配 查询 


ff($bbstype id !=0) { 
S$select->where('bbstype_id =' . Sbbstype id): /指定 查询 类 别 


} 
$select->orWhere($this->getAdapter0->quoteIntoCunhtmlcontent like ?', '%' . $keywords . '%")); /对 内 容 


进行 匹配 查询 
if(Sbbstype id !=0) { 
$select->where('bbstype_id =' . $bbstype_id); /指定 查询 类 别 
} 
} 

机 
Sselect->order($order): // 指 定 查找 结果 排序 方式 
S$adapter = new Zend_Paginator Adapter_DbSelect($select); /指定 分 页 适配器 
Spaginator = new Zend Paginator($adapter): // 生 成 Zend_Paginator 分 页 对 象 


S$paginator->setItemCountPerPage($pageSize)->setCurrentPageNumber($page)->setPageRange($pageRange); 
/指定 分 页 参数 
return $paginator; // 将 Zend_Paginator 对 象 作 为 结果 返回 


} 
< 代码 贴 十 
@ explode0 函 数 用 来 将 字符 串 用 指定 子 串 分 割 ， 并 保存 到 数组 中 。 
@ Sthis->getAdapter()->quoteInto() 方 法 用 来 防止 非法 字符 注入 。 
(3) 在 searchAction 动作 方法 中 通过 tb_title 表 模 型 实例 的 对 象 调用 findByLike() 方 法 来 获取 
Zend_Paginator 对 象 对 查询 结果 进行 分 页 显示 ， 代 码 如 下 : 
$paginator = Sthis-> titleModel->findByLike($keywordsStr, Stypeid, null, Spage. SpageSize): 
(4) 在 searchAction 动作 的 视图 中 ， 通 过 foreach 循环 语句 输出 查询 结果 ， 代 码 如 下 : 


<?php foreach ($this->paginator as $search):?> 
<!-- 省 略 HTML 代 码 ， 用 来 显示 查询 结果 --> 
<?php endforeach:?> 
(5) 为 了 将 查询 结果 进行 分 页 显示 ， 还 需要 建立 分 页 导航 。Zend_Paginator 的 分 页 控制 器 对 象 的 
视图 提供 了 多 个 便于 开发 人 员 建 立 分 页 的 属性 ， 如 表 11.2 所 示 。 


表 11.2 Zend_Paginator 分 页 控制 器 对 象 的 视图 常用 属性 


属 性 说 明 
first | 首页 页 码 号 
firstItemNumber | 当前 页 第 1 条 记录 绝对 编号 


firstPageInRange 分 页 导航 中 首页 页 码 号 
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续 表 
属 性 说 了 明 
current 当前 页 的 页 码 号 
curentItemCount 当前 页 记录 总 数 
itemCountPerPage 每 页 显示 的 记录 数 
last 尾 页 页 码 号 
lastItemNumber 最 后 一 条 记录 号 
lastPageInRange 分 页 导航 中 最 后 一 页 页 码 号 
next 下 一 页 页 码 号 
pageCount 总 页 数 
pagesInRange 根据 所 使 用 导航 条 样式 ， 返 回 页 码 数组 
Previous 上 一 页 页 码 号 
totalItemCount 记录 总 数 


了 解 分 页 控制 器 视图 的 主要 属性 后 ， 下 面 开 始 讲解 建立 分 页 导航 的 步骤 。 首 先 需要 在 search.phtml 
视图 文件 中 所 要 显示 视图 导航 的 位 置 通 过 视图 对 象 的 paginationControl0 方 法 指定 分 页 导航 的 属性 ， 代 
码 如 下 : 


<?php echo $this->paginationControl($this->paginator，'Sliding' ,'searchPaginationControl.phtml', array(keywords' => 
urlencode($this->keywordsStr)))?> 


视图 对 象 的 paginationControl0 方 法 的 语法 格式 如 下 : 
paginationControl(Zend_Paginator paginator, string style, string paginationController, array params) 
参数 说 明 如 表 11.3 所 示 。 

表 11.3 ”视图 对 象 的 paginationControl() 方 法 的 参数 说 明 


属 性 说 阴 
paginator 必要 参数 ， 指 定 Zend_Paginator 对 象 
style 可 选 参数 ， 指 定 导航 条 样式 ， 具 体 样 式 说 明 请 参见 表 11.4 
paginationController 可 选 参数 ， 指 定 分 页 控制 器 视图 文件 名 
Params 可 选 参数 ， 以 数组 形式 指定 要 传递 给 分 页 控制 器 视图 变量 名 称 和 值 
表 11.4 导航 条 样式 说 明 
参数 说 阴 
All 显示 所 有 页 码 链接 
Elastic 类 似 Google 所 使 用 的 导航 条 
Sliding 类 似 Yahoo 所 使 用 的 导航 条 


最 后 还 需要 建立 导航 视图 ，“ 明 日 知道 ”网 站 中 导航 视图 的 代码 如 下 : 


例 程 07 代码 位 置 : 光盘 \TM\11\known\application\modules\default\views\scripts\searchPaginationControl.phtml 
<ul> 
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<li class="lil"> 每 页 &nbsp:<?php echo Sthis->itemCountPerPage?>&nbsp: 条 / 共 查 找到 &nbsp:<?php echo 
S$this->totalltemCount?>&nbsp: 条 </li> 
<liclass="li_cell"></li> 
<?php if($this->totalltemCount>0):?> 
<?php foreach ($this->pagesInRange as Spage):?> 


<Li onclick="javascript:window.location.href='<?php echo 
Sthis->baseUrl(/index/search/keywords/".$this->keywords.'/page/".$page)?>":" class="<?php 
if($this->current—$page):?>li2_curent<?php else:?>1li2<?php endif:?>"><a href="<?php echo 
Sthis->baseUrl(/index/search/keywords/".$this->keywords.'/page/". Spage)?>" class="<?php 


if($this->current—$page):?>a2<?php else:?>al<?php endif:?>"><?php echo $page?></a></li> 
<liclass="li_cell"></li> 
<?php endforeach:?> 
<?php if($this->last > 5):?> 
< onclick="javascript:window.location.hre 人 ='<?php echo 
S$this->baseUrl(/index/search/keywords/'.$this->keywords.'/page/".$this->last)?>":" class="li3"><a href="<?php echo 
S$this->baseUrl(/index/search/keywords/".$this->keywords.'/page/". $this->last)?>" class="al">...<?php echo 
Sthis->last?></a></li> 
<?php endif?> 
<?php endif:?> 
</u> 


用 户 使 用 “明日 知道 ”网 站 查询 内 容 ， 将 在 查询 结果 页 面 显 示 如 图 11.20 所 示 的 分 页 导航 条 。 
nw 站 加 四 加 


图 11.20 多 关键 字 同 时 查询 
11.6.3 ”关键 字 描 红 


使 用 互联 网 搜索 引擎 时 ， 当 用 户 在 搜索 条 中 输入 完 关键 字 并 提交 查询 信息 后 ， 会 在 查询 结果 显示 
页 面 查看 到 所 有 输入 的 查询 关键 字 都 以 红色 显示 ， 这 实质 就 是 关键 字 的 描 红 输出 。 通 过 关键 字 的 描 红 
技术 ， 可 以 使 查询 结果 更 加 醒目 ， 方 便 用 户 进行 进一步 得 选 。“ 明 日 知道 ”网 站 的 查询 结果 描 红 输入 
的 效果 如 图 11.21 所 示 。 


我 有 点 摸 不 着 头绪 ， 朋 友 买 的 编程 词典 我 看 了 一 下 
我 是 今年 刚 工作 的 程序 员 ， 对 一 些 代码 的 运用 还 不 是 很 熟悉 ， 对 代码 的 书写 还 是 不 很 规范 。 公 司 要 我 开发 一 个 项 目 ， 我 有 点 拒 不 着 头绪 ， 


朋友 买 的 编程 词典 我 看 了 一 下 ， 就 得 很 实用 ， 里 面 的 代码 是 编辑 好 的 ， 直 接 就 可 以 实用 ， 省 了 我 不 少 的 时 间 啊 ， 经 理 还 说 我 做 的 块 ， 疆 我 
发 了 奖金， 有 机 会 一 定 要 请 你 们 吃 馈 呢 
wwwmingrizd.com/question/thread.2 10-03-31 02:45 时 日 知 逢 


编程 词典 系列 产品 是 为 编程 者 开发 的 易学 、 易 查 ， 易 用 的 超 媒体 编程 学 习 软件 

编程 词典 系列 产品 是 为 蝙 程 者 开发 的 易 字 、 易 查 、 易 用 的 超 媒 体 注 程 尝 习 软件 ， 它 包 合 技 术 、 人 实例、 方案、 项目、 视频、 界面 、 工 具 等 数 
十 万 开发 资源 。 通 过 编程 词典 ,学习 者 可 以 快速 学 习 编 程 知识 ， 快 速 学 握 编 程 思路 和 方法 ， 并 在 编程 中 体会 快乐 和 乐趣 ， 培养 信心 ， 迅 速 
成 为 屿 程 高 手 。 对 于 开发 者 ， 不 但 可 以 高 效 查 询 相关 技术 ， 快速 异 用 现 有 实例 、 项 目 、 方 案 、 福 码 快速 拱 

www mingrizd. com/question/thread-1 10.03-31 0236 时 日 知 逢 


11.21 查询 结果 的 关键 字 描 红 输 出 


下 面 将 介绍 PHP 中 搜索 模块 关键 字 描 红 输 出 的 制作 方法 。 在 详细 介绍 制作 步骤 前 ， 首 先 介绍 一 下 
制作 描 红 的 基本 思路 : 当 用 户 提 交 查 询 关 键 字 到 搜索 页 面 后 ， 通 过 当前 控制 器 对 象 的 helper 属性 的 
getParam() 方 法 获取 用 户 提交 的 关键 字 ， 然 后 使 用 PHP 提供 的 explode0 函 数 将 提交 关键 字 以 空格 作为 
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分 界 符 保存 到 数组 中 ， 并 将 该 数组 保存 到 视图 变量 中 ， 最 后 定义 一 个 视图 助手 ， 用 于 将 该 数组 中 所 有 
关键 字 描 红 输出 。 具 体 实现 过 程 如 下 : 

(1) 接收 用 户 提交 的 关键 字 ， 并 将 关键 字 保 存 到 数组 中 ， 然 后 将 该 数组 赋 给 视图 变量 ， 以 便 在 视 
图 层 能 够 使 用 该 数组 ， 代 码 如 下 : 


$keywordsStrAIray = explode(' ', $keywordsStr): // 将 关键 字 字 符 串 用 空格 分 隔 保 存 到 数组 中 
S$this->view->assign(keywordsStrArray'` $keywordsStrArray): /将 数组 赋 给 视图 变量 keywordsStrArray 


(2) 在 views 文件 夹 的 helpers 子 文件 夹 下 建立 SetRed.php 文件 ， 在 该 文件 中 建立 用 于 将 关键 字 
描 红 输出 的 自 定义 助手 Zend View_Helper SetRed， 代 码 如 下 : 
例 程 08 ”代码 位 置 光盘 \TM\11\known\application\modules\default\views\helpers\SetRed.php 
class Zend_ View_ Helper SetRed 


public function setRed ($arrayKeywords, $str) 
{ 


SbeginFont = ~—'; // 关 键 字 开始 标识 
SendFont = 呈 // 关 键 字 结束 标识 
foreach ($arrayKeywords as $keyword) { // 人 遍历 所 有 关键 字 
S$array = array(); 
@ preg_match all(/ . $keyword . /i', $str, Sarray): /查询 与 关键 字 匹 配 的 内 容 ， 并 保存 到 数组 中 
SarrayKey = array_unique($array[0]): // 去 掉 重复 数组 元 素 
foreach ($arrayKey as $key) { // 遍 历 所 有 匹配 关键 字 


S$str = str_replace(Skey, $beginFont . Skey . SendFont, $str); // 将 查询 结果 中 匹配 关键 字 前 后 分 别 加 
上 关键 字 开 始 标 识 和 结束 标识 
} 

$str = str_replace($beginFont, '<font color="#FF0000">' $str); // 将 开始 标识 替换 成 <font> 标 签 开始 部 分 

$str = str_replace($endFont, '</font>', $str); /将 结束 标识 替换 成 <fonP> 标 签 结束 部 分 

return $str; 

} 
} 


< 代码 贴 二 
@ preg_match all0 函 数 的 语法 格式 为 preg_match _all(string pattern, string subject array matches, [.int flags]), 该 函数 
用 于 在 subject 中 搜索 与 pattern 给 出 的 正则 表达 式 匹 配 的 内 容 ， 并 将 结果 以 flags 指定 的 顺序 存放 到 matches 中 。 


上 述 所 定义 的 视图 助手 包含 关键 字数 组 和 要 描 红 的 字符 串 两 个 参数 。 该 视图 助手 首先 通过 foreachO 
语句 遍历 关键 字数 组 ， 然 后 使 用 preg_match_all0 函 数 提取 要 描 红 字符 串 中 所 有 与 关键 字 相 匹配 的 字符 
串 ， 并 将 匹配 的 字符 串 保 存在 一 个 数组 中 ， 去 掉 其 中 的 重复 元 素 后 ， 将 该 数组 所 有 元 素 的 开始 和 结束 
加 上 一 个 标识 ， 最 后 将 所 有 的 开始 标识 替换 为 <font color="#FF0000">， 结 束 标识 替换 成 </font>。 最 终 
达到 在 所 要 描 红 字符 串 的 所 有 与 关键 字 匹配 子 串 的 前 后 加 上 <font> 标 签 , 这 样 可 以 达到 在 输出 该 字符 串 
时 ， 字 符 串 中 的 匹配 关键 字 描 红 输 出 的 效果 。 


[| 说 明 : 在 Zend Framework 框架 中 ， 自 定义 视图 助手 一 般 存 放 于 views 目录 的 helpers 子 目录 中 ， 
视图 助手 文件 首 字母 应 该 大 写 , 视图 文件 中 的 类 名 要 以 Zend View_Helper 作为 前 级 ,然后 
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紧 接 与 视图 文件 名 相同 的 字符 作为 完整 类 名 ， 视 图 文件 中 包含 一 个 与 视图 文件 名 相同 的 方 
法 ， 该 方法 首 字母 要 小 写 。 
(3) 在 search.phtml 文件 中 通过 上 面 介绍 的 视图 助手 将 查询 结果 中 的 关键 字 进 行 描 红 输出 ， 代 码 
如 下 了 
echo $this->setRed(Sthis->keywordsStrArray, htmlspecialchars($this->substr($search["unhtmlcontent'], 500))) 


11.6.4 制作 在 线 编辑 器 


以 往 用 HTML 提供 的 文本 域 对 文本 进行 内 容 编辑 时 ， 一 般 是 通过 Javascript 在 要 编辑 文本 的 前 后 
增加 一 些 标记 , 然后 在 输出 这 些 内 容 时 , 将 这 些 标记 替换 成 相应 的 HTML 标记 (如 <font>、<style> 等 ) ， 
这 样 在 显示 内 容 时 ， 就 可 以 出 现 相应 的 特效 。 但 是 ， 这 种 编辑 器 并 不 能 达到 所 见 即 所 得 的 效果 ， 目 前 
已 经 有 很 多 组 织 为 开发 人 员 提 供 了 在 线 文本 编辑 工具 ， 而 且 功 能 也 非常 强大 ，FckEditor 就 是 其 中 比较 
著名 的 在 线 编辑 器 之 一 ， 开 发 人 员 只 需 简 单 的 配置 ， 就 可 以 在 项 目 中 应 用 该 编辑 工具 ， 不 过 这 些 编辑 
器 由 于 功能 较 强大 ， 可 能 存在 访问 页 面 时 加 载 速度 慢 、 针 对 性 不 强 等 缺陷 。 

“明日 知道 ”网 站 在 发 表 和 回复 问题 时 所 使 用 的 在 线 编辑 器 是 通过 JavaScript 编写 的 , 具备 文字 排 
版 、 更 改 文字 样式 、 插 入 文字 链接 和 插入 图 片 等 功能 ， 完 全 适合 论坛 、 博 客 和 后 台 管 理 中 的 内 容 编辑 
工作 ， 图 11.22 为 “明日 知道 ”网 站 所 使 用 的 在 线 编辑 器 。 

ERTAICEETITITCFEICESIEEEELT RICESREE 司 


编程 词 册 系 列 软件 ， 您 贴身 的 编程 专家 ! 上 


图 11.22 在 线 编辑 器 


在 讲解 在 线 编 辑 器 具体 实现 步骤 前 ， 首 先 介绍 制作 一 个 在 线 编辑 器 的 基本 思路 。“ 明 日 知道 ”网 
站 所 使 用 编辑 器 的 编辑 区 域 实质 为 一 个 <ifame> 内 媒 标 签 ， 默 认 情况 下 <iframe> 标 签 是 不 可 编辑 的 ， 需 
要 设置 <iframe> 的 document 属性 的 designMode 属性 的 值 为 on 来 开启 该 标签 的 可 编辑 模式 ， 然 后 使 用 
<iframe> 的 document 属性 的 executeCommand() 方 法 执行 相应 的 命令 标识 即 可 ， 其 中 常用 的 命令 标识 符 
如 表 11.5 所 示 。 


表 11.5 常用 命令 标识 符 


参数 说 了 明 
BackColor 设置 或 获取 当前 选中 区 域 的 背景 颜色 
Bold 切换 当前 选中 区 域 的 粗 体 显示 与 否 


在 当前 选中 区 域 上 插入 超 链接 ， 或 显示 一 个 对 话 框 允许 用 户 指定 要 为 当前 选中 区 域 插入 的 超 链接 的 
URL 
Cut 将 当前 选中 区 域 复制 到 剪贴 板 并 删除 之 


Paste 用 剪贴 板 内 容 覆 盖 当 前 选中 区 域 


CreateLink 
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续 表 
参数 说 明 
Copy 将 当前 选中 区 域 复制 到 剪贴 板 
Delete 删除 当前 选中 区 域 


FontName 设置 或 获取 当前 选中 区 域 的 字体 
FontSize 设置 或 获取 当前 选中 区 域 的 字体 大 小 
ForeColor 设置 或 获取 当前 选中 区 域 的 前 景 〈 文 本 ) 颜色 
Italic 切换 当前 选中 区 域 斜 体 显示 与 否 
JustifyCenter | 将 当前 选中 区 域 在 所 在 格式 化 块 置 中 
JustifyLeft | 将 当前 选中 区 域 所 在 格式 化 块 左 对 齐 
JustifyRight | 将 当前 选中 区 域 所 在 格式 化 块 右 对 齐 
InsertImage | 用 图 像 覆 盖 当 前 选中 区 域 

Undo 取消 

Redo 重 做 

underline 切换 当前 选中 区 域 的 下 划 线 显示 与 否 
strikethrough | 删除 线 


了 解 制作 在 线 编辑 器 的 基本 思路 后 ， 下 面 将 具体 讲解 在 线 编辑 器 的 制作 步骤 。 

(1) 建立 在 线 编辑 器 UI。 在 线 编辑 器 是 通过 JavaScript 实现 的 ， 所 以 将 HTML 代码 作为 字符 串 ， 
然后 使 用 JavaScript 的 document 对 象 的 write0 方 法 输出 到 浏览 器 即 可 。 由 于 代码 所 占 篇 幅 较 长 ， 这 里 
只 给 出 关键 代码 ， 如 下 所 示 。 


例 程 09 ”代码 位 置 : 光盘 \TM\11\known\public_html\js\LzhEditor\LzhEditorjs 
LzhEditor.prototype.Create = function() { 
this.editorStr += "<div id=\"faceLayer\" style=\"position: absolute; width:445px: z-index: 1; border:lpx solid 
#77B7DD; background-color:#f2F9F9: clearboth: display:none:\" onmouseleave=\"this.style.display='none\">"; 
四 /省 略 代 码 请 详 见 本 书 附带 光盘 ， 该 段 代码 主要 用 于 构建 编辑 器 的 控制 面板 
this.editorStr += "<iframe id=\"editor\" name=\"editor\" width=\"100%\" height=\"" 
+ this.height + "\" scrolling=\"auto\" frameborder=\"0\"></ifr ame>": // 通 过 iframe 建 立 编辑 区 域 
this.editorStr += "<input type=\"hidden\" id=\"" + this.fieldName 
+ name=\" +this.fieldName + "\" value=\"\" />": // 通 过 隐藏 域 保存 用 户 在 编辑 器 录入 的 内 容 
this.editorStr += "</div>"; 
document.write(this.editorStr): 
window.frames["editor"].document.open(): // 打 开 ifrfame 
window.frames["editor"].document.write("<BODY style=\"PADDING-RIGHT: Spx: PADDING-LEFT: Spx: 
FONT-SIZE: 12px: PADDING-BOTTOM: 5px: MARGIN: Opx: PADDING-TOP: Spx\">"+ this.value + "</BODY>"); 
// 将 <body> 标 签 写 入 iftame 


window.frames["editor"].document.close(): // 关 闭 i 和 fame 
@ window.frames["editor"].document.designMode = "om": /开启 过 ame 的 编辑 模式 
window.frames["editor"] .focus(): /是 iframe 获 得 焦点 


window.frames["editor"].document.onkeydown = fnnction0 { /为 i 直 ame 添 加 onkeydown 事 件 
if (window.frames["editor"].event.keyCode — 13) { // 如 果 是 回 车 
window.frames['editor'].document.selection.createRange().pasteHTMIL(<br><!---->"): 。” //<br> 后 必须 有 
内 容 才能 换行 ， 所 以 <br> 后 又 加 了 HTML 的 注释 ， 这 样 即 可 以 实现 换行 ， 还 不 会 显示 多 余 内 容 
window.frames['editor].eventretumValue = false: 
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上 
上 
多 
志和 代码 贴 十 
@ 通过 设置 <iffame> 标 签 document 属性 的 designMode 属性 的 值 为 on 来 开启 <iframe> 标 签 的 编辑 模式 。 


上 述 代码 中 ， 使 用 <iframe> 标 签 作为 在 线 编辑 器 的 编辑 区 域 ， 默 认 情 况 下 <ifame> 标 签 是 不 可 编辑 
的 ， 所 以 需要 开启 <iframe> 标 签 的 编辑 模式 。 在 开启 <iframe> 标 签 的 编辑 模式 前 ， 需 要 将 <body> 标 签 写 
入 到 <iframe> 内 部 来 设置 编辑 区 域 的 边框 间距 、 文 字 大 小 等 属性 。 向 <iframe> 中 写 入 <body> 标 签 ， 首 先 
需要 调用 <iframe> 标 签 的 document 属性 的 open0 方 法 打开 <iframe>, 然后 再 通过 该 属性 的 write 方法 将 
<body> 标 签 写 入 到 <iframe> 内 部 ， 最 后 使 用 该 属性 的 close0 方 法 关闭 <iframe> 标 签 。 

(2) 建立 完 在 线 编辑 器 的 身 达 后 ， 下 一 步 就 需要 为 编辑 器 注入 灵魂 。 注 入 灵魂 的 过 程 实质 就 是 让 

在 线 编辑 器 控制 面板 真正 具有 排版 、 设 置 字体 样式 等 功能 ， 即 为 控制 面板 中 各 个 图 标 按钮 的 onclickO 
事件 设置 相应 的 方法 。 为 了 提高 代码 重用 率 ， 这 里 定义 一 个 名 为 LzhEditorFormat0 的 方法 ， 该 方法 中 
有 了 两 个 参数 ， 分 别 为 要 执行 的 命令 标识 〈 即 表 11.5 所 列 出 的 命令 标识 ) 及 执行 该 标识 所 带 的 参数 。 该 
方法 的 代码 如 下 所 示 : 

例 程 10 代码 位 置 ， 光盘 \TM\11\known\public_html\js\LzhEditor\LzhEditorjs 

function lzhEditorFormat(he, pa) { 


‘window.frames["editor"].focus(): // 编 辑 器 获得 焦点 
window.frames["editor"].document.selection.createRange(): // 创 建 编辑 区 域 
i (pa—") { 

window.frames["editor"].document.execCommand(hc, false); // 不 需要 指派 参数 


}else{ 
window.frames["editor"].document.execCommand(hc., false, pa): // 需 要 指派 参数 
} 
5 


通过 上 述 代码 可 知 ， 通 过 <iframe> 标 签 的 document 属性 的 execCommand() 方 法 执行 命令 标识 ， 该 
方法 的 语法 格式 如 下 : 
bSuccess = object.execCommand(sCommand [, bUserInterface] [. vValue]) 
参数 说 明 如 表 11.6 所 示 。 
表 11.6 execCommand() 方 法 的 参数 说 明 


参 数 说 了 明 
sCommand | 必要 参数 ， 所 要 执行 的 命令 标识 
bUserInterface | 可 选 参数 ，boolean 型 ， 默 认为 false， 即 不 显示 用 户 接口 
VValue 可 选 参数 ， 执 行 命令 标识 ， 所 带 参数 。 如 设置 字体 大 小 时 ， 字 号 通过 该 参数 指定 


(3) 通过 上 述 两 步 讲解 ， 相 信 已 经 了 解 制作 一 个 在 线 编辑 器 的 基本 思路 ， 最 后 还 需要 讲解 如 何在 
程序 中 调用 该 编辑 器 。 
由 于 该 编辑 器 被 封装 在 一 个 独立 的 js 文件 中 ， 所 以 首先 应 该 通过 <scripf> 标 签 的 src 属性 包含 该 js 
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文件 。 然 后 在 form 表单 要 显示 编辑 器 的 位 置 使 用 如 下 代码 引入 编辑 器 。 


varlzhEditor= new LzhEditor(name, width. height .value, baseUrl uploadFileUrl); /实例 编辑 器 
lzhEditor.Create0: // 创 建 编辑 器 


通过 上 述 代码 就 可 以 将 编辑 器 显示 在 表单 区 域内 ， 但 当 用 户 提交 表单 前 ， 还 需要 通过 如 下 代码 将 
编辑 器 中 的 内 容 赋 给 隐藏 域 ， 表 单 提交 时 不 能 直接 将 编辑 器 中 的 内 容 提交 给 服务 器 ， 而 是 间接 地 通过 
隐藏 域 实现 ， 代 码 如 下 : 

lzhEditor.Submit(): // 将 编辑 器 中 内 容 赋 给 隐藏 域 


以 上 就 是 制作 一 个 在 线 编辑 器 的 全 部 过 程 ,最 后 再 总 结 其 开发 思路 : 首先 开启 <iframe> 标 签 的 编辑 
模式 ， 然 后 通过 execCommand0 方 法 执行 对 应 的 命令 标识 即 可 对 <iframe> 中 的 文本 进行 编辑 。 


11.7 MVC 框架 结构 搭建 


“明日 知道 ”网 站 采用 Zend Framework 框架 技术 开发 ， 所 以 首先 应 该 设计 并 搭建 出 稳定 、 扩 展 性 
强 的 框架 结构 。 为 了 能 够 使 读者 掌握 Zend Framework 框架 的 搭建 方法 ， 下 面 通过 一 个 简单 的 实例 对 此 
进行 介绍 。 


11.7.1 ”MVC 开发 模式 介绍 


MVC 分 别 为 英文 单词 model、view 和 controller 的 首 字母 的 大 写 , 中 文 含义 为 “模型 -视图 -控制 器 ”， 
这 种 开发 模式 是 现 阶段 主流 的 项 目 开 发 模式 之 一 ， 可 使 整个 项 目的 各 个 层次 独立 出 来 ， 这 样 程序 开发 
人 员 和 美工 人 员 可 以 对 项 目 进 行 同步 开发 ， 而 且 也 为 项 目 进一步 的 功能 扩展 和 日 后 在 生产 环境 下 的 维 
护 工作 提供 了 极 大 方便 ， 所 以 MVC 开发 模式 也 是 现 阶段 各 大 软件 公司 极力 提倡 的 开发 模式 ， 是 程序 
开发 人 员 的 必修 内 容 。 图 11.23 为 典型 的 MVC 模式 架构 图 。 


模型 
通知 视图 改变 
应 用 程序 功能 
响应 状态 查询 
封装 应 用 程序 状态 


控制 器 


允许 控制 器 选择 视图 用 户 动作 映射 成 模型 更 新 
发 送 用 户 输入 给 控制 器 定义 应 用 程序 行为 
解释 模型 请 选择 响应 的 视图 


11.23 MVC 模式 架构 图 
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11.7.2 ”Zend Framework 框架 的 MVC 目录 结构 


日 多 appication 
-BB configs 
HB forms 
-BS layouts 
mB models 
日 铝 modules 

日 久 defauk 


controlers 
EB vews 


© flers 
-Er helpers 
日 久 scripts 


日 全 dnin 
由 仿 controllers 
DB vews 


© fkers 
By helpers 
由 舍 srpts 


由 它 plugins 


由 - 它 resources 
由 - 国 bootstrap,php 


日 全 public_html 


HB upfies 
国 htaccess 
田 -|@ ndex,php 


11.24 “明日 知道 ”网 站 框架 结构 


项 目 引 导 文 件 


# 开 启 URIL 重 写 
RewriteEngine on 


# 除 扩展 名 为 js、.css、.gif、.jpg、.png、.bmp 的 文件 外 ， 访 问 其 他 文件 都 转向 到 index.php 引 导 文件 
RewriteRule \\.(jslcsslgifljpglpnglbmp)$ index.php 


与 其 他 主流 的 PHP 框架 相 比 ，Zend Framewok 是 一 款 相对 灵活 的 MVC 框架 ， 其 搭建 方式 有 多 种 ， 
其 中 “明日 知道 ”网 站 的 MVC 框架 结构 如 图 11.24 所 示 。 从 图 中 可 以 看 出 , 该 结构 是 一 种 多 模块 结构 ， 
即将 前 后 和 后 台独 立成 两 个 模块 ， 从 而 更 有 利于 多 人 协作 开发 。 


11.7.3 Zend Framework 框架 的 MVC 结构 创建 过 程 
了 解 “明日 知道 ”网 站 的 框架 结构 图 后 ， 下 面具 体 按 步 骤 讲 解 使 用 Zend Framework 建立 多 模块 


MVC 框架 结构 的 搭建 过 程 。 为 了 便于 学 习 ， 这 里 给 出 一 个 最 基本 的 Zend Framework 框架 的 流程 ， 如 
图 11.25 所 示 ， 之 后 开发 人 员 就 可 以 在 此 基础 上 进行 扩充 。 


| 创建 视图 文件 index.phtml 


[运行 一 个 最 基本 的 ZF 程序 


图 11.25 最 基本 的 Zend Framework 程序 开发 步 又 
(1) 在 public_html 目录 下 建立 URL 重 写 文件 .htaccess， 并 在 该 文件 中 输入 URL 重 写 规则 ， 如 下 
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[| 说 明 : 由 于 .htaccess 文件 没有 文件 主 名 ， 所 以 在 Windows 系统 下 无 法 直接 命名 ， 下 面 介绍 两 种 用 
户 创建 .htaccess 文件 的 方法 。 
方法 一 : 通过 Windows 系统 的 copy con 命令 创建 ， 如 图 11.26 所 示 。 创 建 完成 后 使 用 Ctrl+Z 键 退 
出 编辑 模式 。 


C:\WINDOWS\system32\cmd.exe = 上 | xj 
< ht 站 


图 11.26 通过 命令 提示 符 创建 .htaccess 文件 


方 ; 通过 “记事 本 ”文本 编辑 工具 将 文件 另存 为 文件 名 为 .htaccess 的 文件 ， 这 里 应 该 注意 保 
存 文 件 时 需要 选择 文件 类 型 为 所 有 文件 。 
(2) 在 public_html 目录 下 创建 mdex.php 引导 文件 ， 代 码 如 下 : 


defined(APPLICATION PATH') || define(APPLICATION PATH,', realpath(dimame( FILE_ ) . "../application")); 
// 应 用 路 径 

defined(APPLICATION ENV") 中 define('APPLICATION ENV., getenv(APPLICATION ENV') 
getenv('APPLICATION_ENV") : 'project’): // 应 用 环境 

$arrayIncludePath = array("., realpath(dirname( FILE _) .'/../library")): /指定 工程 包含 目录 

set_include_path(implode(PATH_SEPARATOR., $arrayIncludePath)); 。“ // 将 指定 路 径 包 含 到 工程 中 

require_once 'Zend/Application.php'; // 包 含 Application.php 文 件 

$application = new Zend_Application(APPLICATION_ENV, APPLICATION PATH . '‘/configs/application.ini’); /实例 
Zend_Application 类 

$application->bootstrapO->run(): // 调 用 启动 文件 并 运行 项 目 


上 述 代 码 中 ,首先 定义 两 个 常量 APPLICATION PATH 和 APPLICATION_ENV， 分 别 用 来 保存 应 
用 路 径 和 应 用 环境 名 ， 这 样 在 工程 的 其 他 模块 中 就 可 以 直接 使 用 。 然 后 将 当前 目录 地 址 “.” 和 Zend 
Framework 类 库 地 址 作为 数组 元 素 保存 在 名 为 $arrayIncludePath 的 数组 中 , 并 使 用 implode0 函 数 将 上 述 
路 径 用 PATH_SEPARATOR 常量 连接 ， 最 后 使 用 set_include_path0 方 法 将 路 径 导 入 到 工程 中 ， 从 而 在 
工程 中 就 可 以 直接 找到 这 些 路 径 下 的 文件 。 

Zend Framework 类 库 被 导入 到 工程 中 之 后 ,就 可 以 使 用 require_once 等 文件 包含 语句 包含 Zend 目 
录 下 的 Application.php 文件 ， 之 后 实例 Zend_Application 类 ， 并 通过 引用 该 类 的 bootstrap0 方 法 调用 启 
动 类 ， 最 终 再 通过 调用 run() 方 法 运行 工程 。 

(3) 在 第 (2) 步 实 例 Zend_Application 类 时 ， 为 类 的 构造 函数 传 入 了 一 个 名 为 application.ini 的 

文件 ， 该 文件 即 为 工程 的 配置 文件 ， 本 步 将 介绍 该 文件 的 创建 过 程 ， 其 中 一 个 基本 的 配置 文件 如 下 所 示 ; 

[project] 

# 设 置 错误 级 别 

phpSettings.display_startup_errors = 1 


phpSettings.display_errors= 1 
# 设 置 时 区 
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phpSetting.date.timezone ='Asia/Shanghai' 

# 配 置 启动 类 

bootstrap.path = APPLICATION PATH "/Bootstrap.php" 

bootstrap.class = "Bootstrap" 

# 配 置 工程 模块 

Tesources.frontController.moduleDirectory = APPLICATION PATH "/modules" 

配置 文件 存放 位 置 和 名 称 并 不 固定 ， 只 要 与 实例 Zend_Application 类 传递 的 参数 一 致 即 可 。 

(4) 在 application.ini 配置 文件 中 指定 启动 类 的 位 置 后 ， 在 工程 中 就 可 以 找到 启动 文件 Bootstrap 

.Php， 其 关键 代码 如 下 : 


clclass Bootstrap extends Zend_Application Bootstrap_ Bootstrap 


{ 
// 基 本 的 启动 类 可 以 不 进行 任何 操作 
EL 
从 上 述 代码 可 知 ， 启 动 Bootstrap 继承 自 类 Zend_Application_Bootstrap_Bootstrap, 该 类 已 经 实现 最 
基本 的 启动 配置 工作 ,所 以 在 启动 类 中 不 需要 包含 任何 代码 就 可 以 运行 最 基本 的 Zend Framework 应 用 。 
(5) 完成 以 上 操作 步骤 后 ， 就 可 以 创建 默认 控制 器 IndexController， 同 样 一 个 最 基本 的 控制 器 应 
该 包含 一 个 首页 动作 indexAction， 代 码 如 下 : 


class IndexController extends Zend_Controller Action 


public function indexAction 0 /默认 动作 


Sthis->view->testStr="Hello ZF!": /为 视图 变量 赋值 


} 
} 
[a 说 明 : 使 用 Zend Framework 时 ， 控 制 器 名 应 以 Controller 结束 ， 动 作 名 应 以 Action 结束 。 


(6) 完成 默认 控制 器 及 首页 动作 的 创建 后 ， 就 可 以 创建 视图 文件 index.phtml 了 ， 为 了 查看 效果 ， 
在 视图 文件 中 仅 输出 首页 动作 中 指定 的 视图 变量 的 值 ， 代 码 如 下 : 


echo S$this->testStr: 


(7) 假设 上 述 工程 所 在 的 目录 为 Apache 默认 主 目录 的 test 子 目录 ， 则 可 以 通过 如 下 的 URL 进行 

访问 : 

http://localhost/test/public_html。 

http://localhost/test/public_html/index。 

http://localhost/test/public_html/index/index。 

正常 情况 下 ，Zend Framework 的 访问 URL 应 该 为 “域名 + 模块 名 + 控制 器 名 + 动作 名 ”， 那 么 为 什 
么 上 述 URL 省 略 了 部 分 内 容 也 可 以 访问 呢 ? 这 是 因为 在 Zend Framework 中 ， 默 认 模 块 名 、 默 认 控 制 
器 名 和 首页 动作 名 可 以 省 略 。 

运行 上 述 工 程 ， 可 以 在 浏览 器 中 输出 “Hello ZF!”。 
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11.8 首页 设计 


网 站 首页 的 内 容 应 该 突出 网 站 主题 及 特色 ， 这 点 在 网 站 的 生命 周期 中 起 着 非常 重要 的 作用 。 本 节 
将 以 “明日 知道 ”网 站 为 例 讲解 如 何 设计 合理 的 、 适 合 网 站 特点 的 首页 。 


11.8.1 首页 概述 


设计 网 站 首页 要 贴切 网 站 主题 、 突 出 网 站 特色 ， 而 非 复杂 就 可 以 。 考 虑 “明日 知道 ”网 站 主要 用 
于 为 编程 相关 工作 人 员 提 供 技术 问题 搜索 和 技术 交流 , 因此 开发 人 员 借鉴 互联 网 一 些 知名 的 搜索 工具 ， 
设计 出 如 图 11.27 所 示 的 首页 界面 。“ 明 日 知道 ”网 站 的 首页 主要 由 网 站 Logo、 语 言 导航 条 、 搜 索 区 、 
友情 链接 和 版 权 声明 等 部 分 组 成 ， 页 面 简洁 大 方 、 突 出 核心 ， 便 于 用 户 使 用 。 


于 | 注册 


| 
Ce Li ASPNET HP 


版 权 声明 明日 图 世 | 明日 软件 | 编程 词 上 
园 丰 服务 热线 : 400.6751066 仿真 : 0431-34939777 人 kp 补 : mingrisoft@min| 


公司 地 址 ; 吉林 省 发 春 市 一 道 区 本 县 大 衡 39 寻 亚 泰 广场 C 座 。 言 CP 备 07500073 导 
Copynght © www merbook com Al Rights Reservedl 


图 11.27 明日 知道 网 站 首页 


11.8.2 ”首页 技术 分 析 


“明日 知道 ”网 站 首页 的 制作 主要 涉及 两 点 较为 关键 的 技术 ,分 别 为 类 似 Google 搜索 条 的 制作 和 
语言 类 别 选项 卡 的 制作 ， 其 中 类 似 Google 搜索 条 的 制作 过 程 已 经 在 本 章 技术 攻关 中 讲解 过 了 ， 这 里 不 
再 袭 述 ， 下 面 重点 对 语言 选项 卡 的 实现 过 程 进行 技术 分 析 。 

与 Visual Basic、C# 等 一 些 可 视 化 编程 语言 相 比 ， 网 页 视图 层 的 一 些 组 件 和 特效 的 制作 需要 通过 编 
码 实现 ， 而 非 简单 地 拖 搜 ， 这 样 制作 出 的 组 件 更 具 灵 活性 。 制 作 页 面 特效 一 般 通 过 JavaScript、Ajax 等 
技术 实现 。 

制作 “明日 知道 ”网 站 语言 类 别 选项 卡 的 基本 思想 是 : 首先 从 数据 库 中 提取 所 要 显示 语言 类 别 的 
名 称 ， 并 通过 <div> 标 签 显示 在 同一 行 ， 然 后 为 这 些 <div> 标 签 指定 背景 图 片 ， 最 后 定义 如 下 方法 〈 这 
里 用 jQuery 实现 ) 控制 选项 卡 之 间 的 切换 。 
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例 程 11 代码 位 置 : 光盘 \IM\11\known\application\modules\default\views\scripts\index\index.phtml 
function changeSearchType(x, y) 
{ /该 方法 中 的 两 个 参数 x，y 分 别 为 每 个 选项 卡 按钮 的 D 和 选项 卡 对 应 语言 的 ID 
S$("#st_0").css("background", "url(<?php echo Sthis->baseUrl(/img/main dh btm2.gif)?>)"): /更 改 全 部 选项 卡 背 景 


图 片 
S$("#st_0").css("color", "#333333"): // 更 改 全 部 选项 卡 前 景色 
<?php foreach (S$this->types as $type):?> 
S$("#st <?php echo Stype['tid]?>").css("background", "url(<?php echo 
Sthis->baseUrl(/img/main_dh btn2.gif)?>)"): // 更 改 语言 类 别 选项 卡 背景 图 片 
$("#st_<?php echo $type['tid]?>").css("color", "#333333"): /更 改 语言 类 别 选 项 卡 前 景色 
<?php endforeach:?> 


S$("#"+x).css("background", "url(<?php echo S$this->baseUrl(/img/main dh btnl.gif)?>)"); /更 改 当 前 选中 的 选 
项 卡 背 景 图 片 


S$("#"+x).css("color", "HEFFFFF"); /更 改 当 前 选中 的 选项 卡 的 前 景色 
S("#typeid").val(y); // 更 改 隐藏 域 typeid 的 值 为 当前 选中 选项 卡 对 应 的 语言 DD 


} 


上 述 代码 中 ， 通 过 jQuery 的 css0 方 法 更 改 <div> 标 签 的 背景 图 片 和 前 景色 ， 该 方法 可 以 传 入 两 个 
参数 ， 第 1 个 参数 为 要 更 改 的 CSS 样式 的 属性 名 称 ， 如 color、background-color、font-size 等 ; 第 2 个 
参数 为 为 属性 所 指定 的 值 。 其 中 第 2 个 参数 可 以 省 略 ， 如 果 设 置 第 2 个 参数 ， 则 需要 为 CSS 样式 重新 
赋值 ， 否 则 将 获取 该 CSS 样式 的 值 。 


11.8.3 ”首页 的 实现 过 程 


“明日 知道 ”网 站 采用 MVC 开发 模式 ， 所 以 其 开发 过 程 也 需要 按 模 型 层 、 视 图 层 和 控制 器 层 进 
行 。 在 对 网 站 首页 进行 技术 分 析 时 已 经 介绍 到 ， 制 作 该 网 站 首页 需要 从 数据 库 中 提取 所 有 语言 类 别 ， 
所 以 应 该 在 类 别 表 的 模型 Model_DbTable_Bbstype 中 建立 一 个 用 来 查询 全 部 类 别 的 方法 ， 这 里 命名 为 
findAl10， 其 代码 如 下 : 

例 程 12 ”代码 位 置 : 光盘 \TM\11\known\application\models\DbTable\Bbstype.php 

public function findAll ($isCache = false) 


SinnerSelect] = $this->getAdapter()->selectO: // 所 有 主题 

SinnerSelect1->from('tb_title'. ‘count(*)")->join(S$this-> name. 'tb title.bbstype id=' . S$this-> name . ‘id', 
null)->where($this->_name . ‘id = tid’): 

SinnerSelect2 = $this->getAdapter()->selectO: // 今 日 主题 

SinnerSelect2->from('tb_title'. 'count(*)'")->join(S$this-> name, ‘tb title.bbstype id- . S$this-> name . ‘id', 
null)->where('date(tb_title.addtime)=date(now(O)")->where(S$this-> name . "id = tid"); 

SinnerSelect3 = $this->getAdapter()->selectO: // 最 后 主题 ID 


SinnerSelect3->from('tb_title', ‘tb_title.id)->join(Sthis-> name. 'tb title bbstype id=" . Sthis-> name . ‘id, 
null)->where($this-> name . ‘id = tid")->order('tb_title.addtime desc)->limit(1. 0): 

SinnerSelect4 = Sthis->getAdapter0->select0: /最 后 主题 

SinnerSelect4->from('tb title', ‘tb titletitle)->join($this-> name. ‘tb titlebbstype id='" . S$this-> name . ‘id', 
nulD)->where($this-> name . '.id = tid")->order('tb_title.addtime desc)->limit(1. 0): 

SinnerSelect5 = $this->getAdapter()->select(): // 最 后 主题 时 间 

SinnerSelect5->from('tb_title'. tb_title.addtime")->join($this-> name. 'tb_title.bbstype id- . $this-> name . ‘id', 
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null)->where($this-> name .'.id = tid")->order('tb_title.addtime desc)->limit(1. 0): 

SinnerSelect6 = $this->getAdapter()->select(: // 总 回复 

SinnerSelect6->from('tb reply’, ‘count(*)'")->join($this-> name， 'tb_ reply.bbstype id=" . Sthis-> name . ‘id', 
null)->where($this-> name . "id= tid"): 

SinnserSelect7 = $this->getAdapter()->selectO|: /最 后 回复 用 户 ID 

SinnserSelect7->from('tb title，'b titleuser id)->join($this-> name，'tb _title.bbstype id- . $this-> name . ‘id', 
null)->where($this-> name . 'id=tid)->order('tb title.addtime desc)->limit(1. 0): 


S$innserSelect8 = S$this->getAdapter()->select(|): /最 后 回复 用 户 昵称 
SinnserSelect8->from('tb_user', metname')->where('id=( . SinnserSelect7 . )): 
S$select = $this->getAdapter()->select|:; // 外 层 查询 


Sselect->from($this-> name, array($this-> name . ‘id as tid , S$this-> name . ‘typename'’ . S$this-> name . 
".description' , $this-> name . .addtime' , '( . SinnerSelectl . ) as totaltitle' , '( . $innerSelect2 . ) as totaltodaytitle’ , '(' . 
SinnerSelect3 . ) as lasttitleid' , "(' . $innerSelect4 . ) as lasttitletitle' , '(' . SinnerSelects . ) as lasttitletime' , '(' . $innerSelect6 . ) 
as totalreply' , "(' . $innserSelect8 . ) as lasttitlenetname'))->order(" . Sthis-> name . '.addtime asc'); 

if ($isCache) { // 如 果 缓 存 查询 结果 

让 (! $result = $this-> cache->load(strtoupper($this-> name .' findAll))) { 
// 判 断 缓存 文件 是 否 存在 或 缓存 是 否 过 期 
$result = $this->getAdapter0->fetchAll($selecb:// 如 果 缓 存 文 件 不 存在 或 缓存 过 期 ， 则 重 写 查找 
S$this->_cache->save($result, strtoupper($this-> name .' findAll)): 


// 写 入 并 生成 新 缓存 
} 
}else { 
Sresult = $this->getAdapter()->fetchAll($select); /不 缓存 则 直接 查找 
return $result: // 返 回 结果 


} 


上 述 代 码 中 查询 全 部 语言 类 别 使 用 Zend_Select 及 其 子 查询 和 Zend_Cache 缓存 技术 实现 。 

下 面 首先 介绍 如 何 使 用 Zend_Select 进行 查询 。 使 用 Zend_Select 进行 查询 ， 需 要 获得 数据 库 适 配 
器 对 象 ， 然 后 通过 该 对 象 的 select0 方 法 生成 Zend_Select 对 象 ， 如 下 所 示 : 

S$select = $this->getAdapter()->select(: 


其 中 , $this 代表 当前 表 的 模型 , 由 于 该 模型 继承 自 Zend_Db_Table 类 , 所 以 通过 $this->getAdapterO 
方法 即 可 获取 数据 库 适 配器 。 获 得 Zend_Select 对 象 后 ， 就 可 以 通过 该 对 象 的 fom() 方 法 指定 查询 的 表 
名 、 字 段 名 。 通 过 order0 方 法 指定 记录 排序 方式 。 

那么 如 何 通过 Zend_Select 进行 子 查询 呢 ? 其 原理 是 : 首先 定义 一 个 Zend_Select 对 象 作为 子 查询 ， 
然后 在 外 层 查 询 中 ， 将 该 Zend_Select 对 象 作 为 查询 结果 的 一 个 字段 出 现 ， 代 码 如 下 : 

SinSelect = $db->selectO: /内 层 查询 ， 这 里 只 给 出 关键 代码 


SoutSelect = $db->select0: // 外 层 查询 
S$outSelect->from( 表 名 , array( 字 段 1. 字段 2.(.SinSelect) as 字段 3')): // 指 定 外 层 查 询 表 名 和 字段 名 


最 后 介绍 如 何 对 查询 结果 进行 缓存 输出 。 在 Zend Framework 框架 中 ， 使 用 Zend_Cache 对 数据 进 
行 缓存 来 提高 程序 的 执行 效率 。 在 制作 上 述 查 询 全 部 语言 类 别 方法 时 ， 为 了 减少 对 数据 库 频繁 查询 的 
次 数 ， 降 低 数据 库 服务 器 负载 ， 提 交 程 序 执行 效率 ， 这 里 使 用 Zend_Cache 对 查询 结果 进行 缓存 。 
Zend_Cache 通过 其 静态 方法 factory0 获 得 其 实例 ， 代 码 如 下 : 
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$cache = Zend_Cache::factory(string frontendName. string backendName. array fiontendOptions. array backendOptions): 
参数 说 明 如 表 11.7 所 示 。 
表 11.7 Zend_Cache 的 factory() 方 法 的 参数 说 明 
属 性 值 说 了 明 
必要 参数 ， 缓 存 前 端 名 称 
必要 参数 ， 缓 存 后 端 名 称 
， 缓 存 前 端 选项 
， 缓 存 后 端 选 项 


frontendName 
backendName 


frontendOptions 


为 了 更 加 明确 地 了 解 Zend_Cache 的 使 用 方法 ， 下 面 以 本 项 目 为 例 分 步 讲解 Zend Framework 中 组 
存 的 使 用 方法 。 
(1) 为 了 便于 以 后 维护 ,将 缓存 过 期 时 间 和 缓存 文件 路 径 在 application.ini 配置 文件 中 指定 ,代码 
如 下 Fs 
# 缓 存 时 间 
cache.leftTime = "3600" 


# 级 存 文件 存储 目录 
cache.cache dir = APPLICATION PATH "/tmp/" 


(2) Zend_Cache 对 象 应 该 被 项 目 全 局 应 用 ， 所 以 该 对 象 在 启动 文件 中 被 初始 化 ， 代 码 如 下 : 


Sconfig = $this->getOptions0: ”// 通 过 启动 类 的 getOptions0 方 法 获取 配置 文件 配置 信息 ， 并 保存 在 多 维 数组 中 返回 
Zend Registry::set('config' Sconfig): /注册 该 多 维 数组 

$frontOptions = aray(leftTime' => $config['cache'][leftTime'] , 'automatic_serialization' => true); 

/缓存 前 端 选项 

$backOptions = array('cache dir => $config['cache]['cache_dir]): /缓存 后 端 选项 

Scache = Zend_Cache::factory('Core', 'File', $frontOptions, SbackOptions): /获取 Zend_Cache 对 象 
Zend_Registty::set(cache', Scache); /注册 Zend_Cache 对 象 


(3) 使 用 Zend_Register 的 静态 方法 set0 将 缓存 对 象 注册 后 ， 就 可 以 在 工程 的 其 他 模块 通过 Zend 
_Register 的 get() 方 法 获取 缓存 对 象 并 使 用 。 
private $_cache: /将 缓存 对 象 声 明 为 私有 成 员 ， 这 样 就 可 以 在 模型 的 其 他 方法 中 使 用 
public function init () 


S$this-> cache = Zend_ Registry::get(cache): /获取 缓存 对 象 
} 
(4) 在 类 别 表 模 型 的 findAl10 方 法 中 ， 通 过 缓存 对 和 象 的 load0 方 法 判断 在 缓存 目录 下 是 否 有 以 该 
方法 参数 为 缓存 ID 的 缓存 文件 并 且 不 过 期 。 如 果 存 在 ， 从 缓存 文件 中 获取 要 查询 的 内 容 ; 否则 ， 从 数 
据 库 中 查询 内 容 并 保存 在 缓存 文件 中 ， 代 码 如 下 : 
f(! Sresult = $this->_cache->load(strtoupper($this-> name .' findAll))) {/ 判 断 缓 存 文件 是 否 存在 或 缓存 是 否 过 期 


Sresult = $this->getAdapter()->fetchAll(Sselect): // 如 果 缓 存 文件 不 存在 或 缓存 过 期 ， 则 重 写 
查找 
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S$this-> cache->save(Sresult, strtoupper(Sthis-> name .' findAll)): V/ 写 入 并 生成 新 缓存 


11.9 用 户 模块 设计 
用 户 作为 网 站 活动 的 参与 者 和 维护 者 ， 是 网 站 能 够 生存 发 展 的 源 动 力 。“ 明 日 知道 ”网 站 由 于 需 
要 用 户 提 出 问题 、 解 决 问题 ， 所 以 制定 合理 的 用 户 模块 显得 十 分 重要 。 
11.9.1 用户 模 块 概述 
“明日 知道 ”网 站 的 用 户 模块 主要 包括 用 户 注册 、 登 录 、 用 户 信息 管理 等 功能 。 用 户 如 果 只 进行 
问题 查找 ， 则 无 须 注册 ， 如 果 没 有 在 该 站 点 查找 到 预期 的 结果 ， 那 么 可 以 将 问题 发 布 到 “明日 知道 ” 


网 站 中 ， 其 他 用 户 就 可 以 一 起 探讨 、 解 答 ， 这 时 就 需要 用 户 注 册 为 本 站 会 员 。 用 户 访问 “明日 知道 ” 
网 站 的 流程 如 图 11.28 所 示 。 


11.28 用户 访问 流程 
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11.9.2 ”用 户 模块 技术 分 析 


为 了 防止 非法 用 户 通 过 恶意 程序 ， 采用 试探 密码 的 方式 登录 本 站 , 或 连续 地 向 用 户 表 中 注入 信息 ， 
“明日 知道 ”网 站 的 用 户 登 录 和 注册 表单 中 都 增加 了 验证 码 ， 从 而 可 以 有 效 地 预防 上 述 情况 发 生 。 

在 Web 项 目 中 ， 验 证 码 可 以 有 多 种 形式 ， 如 文字 验证 码 、 图 片 验 证 码 、 语 言 验证 码 等 ， 本 站 所 采 
用 的 是 Web 项 目 中 最 为 常用 的 验证 码 形式 ， 即 图 片 验证 码 ， 如 了 
图 11.29 所 示 。 a 

使 用 PHP 语言 制作 图 片 验证 码 可 以 使 用 GD2 函数 绘制 。 考 一 
虑 到 验证 码 的 高 度 、 宽 度 以 及 图 片 中 文字 的 字体 在 不 同类 型 的 表 本 
单 中 可 能 存在 差异 ， 在 制作 “明日 知道 ”网 站 验证 码 时 ， 单 独 将 图 11.29 图 片 验证 码 
验证 码 的 实现 过 程 封装 成 验证 码 类 。 那 么 ， 验 证 码 类 是 如 何 实现 的 呢 ? 下 面 对 此 进行 讲解 。 

(1) 将 该 验证 码 类 命名 为 Plugin_Util ValidateCode， 这 样 命名 的 原因 是 如 果 将 该 验证 码 类 保存 在 
application 目录 下 的 util 子 目录 中 ， 并 将 该 文件 命名 为 ValidateCode. php， 那么 在 需要 使 用 该 验证 码 类 
时 ， 根 据 Zend Framework 的 设计 特点 ， 不 需要 使 用 include 等 包含 语句 ， 就 可 以 直接 使 用 。 

定义 该 验证 码 类 的 数据 成 员 ， 代 码 如 下 所 示 : 
例 程 13 ”代码 位 置 ， 光 扒 \TM\11\known\public_html\known\application\plugins\Util\ValidateCode.php 


Private $_width; // 宽 度 

Private $_height: /高 度 

Private $_codeStr; /验证 码 

private $_fontType: // 设 置 字体 类 型 0 表示 粗 体 1 表示 简体 
Private $_imsg; // 图 像 句柄 


(2) 定义 验证 码 类 的 构造 方法 ， 实 现 验 证 码 类 的 初始 化 ， 代 码 如 下 : 


例 程 14 ”代码 位 置 ; 光盘 \TM\11\known\public_html\known\application\plugins\Util\ValidateCode.php 
public function _construct ($width, $height, $codeStr = '0000', $fontType = 0) 
{ 


Sthis->_width = $width: /验证 码 宽度 
Sthis-> height = Sheight /验证 码 高 度 
S$this->_ codeStr = substr($codeStr, 0. 4): /验证 码 的 值 
S$this-> fontType = $fontType: /字体 


1 


(3) 定义 类 的 私有 方法 _getColor0， 用 于 生成 颜色 句柄 ， 代 码 如 下 : 


例 程 15 代码 位 置 光盘 \TM\11\known\public_html\known\application\plugins\Util\ValidateCode.php 


Private function getColor ($x, $y) 
| 


S$r = mt_rand($x, $y): // 红 
Sg = mt rand($x, $y): // 绿 
$b = mt rand($x. $y): // 蓝 
return imagecolorallocate($this-> img, Sr $g, Sb): // 返 回 颜色 句柄 
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(4) 定义 私有 方法 init0， 用 于 创建 图 片 验 证 码 对 象 ， 代 码 如 下 : 


例 程 16 ”代码 位 置 : 光盘 \TM\11\known\public_ html\known\application\plugins\Util\ValidateCode.php 
private function init 0 


Sthis-> img = imagecreate($this-> width, Sthis-> height); 。“// 创 建 图 像 


(5) 在 创建 的 验证 码 图 片 对 象 的 基础 上 创建 验证 码 体 ， 包 括 设置 验证 码 的 填充 色 、 边 框 ， 验 证 码 
上 文字 内 容 及 字体 等 ， 其 实现 代码 如 下 : 


例 程 17 ”代码 位 置 光盘 \TM\11\known\public_html\known\application\plugins\Util\ValidateCode.php 
Private function _build 0 

imagefill($this-> img, 0, 0, $this-> getColor(150, 250)): // 为 图 像 填充 背景 色 
imagerectangle($this-> img, 0, 0, $this-> width - 1, $this-> height - 1, $this-> getColor(50, 150)); 
/创建 一 个 矩形 作为 验证 码 的 边框 


if (S$this-> fontType 一 0){ /设置 验证 码 文字 的 字体 
SfontFileName = 'ARIALBI.TTF'; 
} else { 
SfontFileName ='ARIALN. TTF'; 
b 
for ($i= 0; $i < strlen($this-> codeStD: $i++) { /绘制 文字 
0 imagettftext($this-> img, mt rand(12. 24). 0, ($this-> width) / 4 * $i, mt rand(20, S$this-> height - 5), 
S$this->_getColor(10, 180), APPLICATION PATH . Wresources/ . $fontFileName, substr($this->_ codeStr. $i, 1)); 
for ($i=0; $i< 15: Si++) { // 绘 制 15 条 干扰 线 
© imageline($this-> img, mt rand(0, $this-> width), mt rand(0, S$this-> height), mt rand(0, $this-> width), 
mt_rand(0, $this-> height), $this->_getColor(110, 210)); 
} 
} 
Ah 代码 贴 二 


@ imagettftext0 方 法 : 用 于 按 指定 的 字体 输出 字符 囊 。 
@ imageline() 方 法 : 用 于 绘制 直线 
(6) 创建 show() 方 法 用 于 显示 图 片 ， 代 码 如 下 : 


例 程 18 代码 位 置 : 光盘 \TM\11\known\public_html\known\application\plugins\Util\ValidateCode.php 


public function show 0 

{ 
header('content-type:image/pne"): // 设 置 输出 图 片 格式 
Sthis-> init0: /图 片 初始 化 
Sthis-> build0: /建立 图 片 体 
imagepng($this-> img): // 输 出 图 片 

上 
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11.9.3 ”用户 注册 的 实现 过 程 


通过 用 户 注册 ， 可 将 用 户 的 基本 信息 记录 下 来 ， 便 于 用 户 参与 站 内 活动 及 用 户 之 间 的 交流 。 


“ 明 


日 知道 ”网 站 的 注册 表单 如 图 11.30 所 示 ， 该 表单 主要 包括 用 户 的 昵称 、 登 录 密 码 、E-mail 地 址 、 联 系 


电话 和 所 在 地 等 信息 。 


确认 容 码 , 


图 11.30 用 户 注册 表单 
了 解 注册 表单 的 结构 后 ， 下 面 讲解 该 表单 的 实现 过 程 。 


(1) 建立 用 户 注册 表单 。“ 明 日 知道 ”网 站 的 注册 表单 是 通过 Zend Framework 的 表单 视图 助手 


实现 的 ， 并 通过 jQuery 技术 对 表单 数据 进行 效 验 。 注 册 表 单 UI 的 实现 代码 请 详 见 本 书 附 带 光 盘 。 


(2) 保存 用 户 注册 信息 ， 并 向 注册 用 户 的 邮箱 中 发 送 注册 成 功 提 示 。 当 用 户 填 写 完 注 册 信 息 ， 并 


提交 注册 表单 后 ， 信 息 将 被 提交 到 用 户 控制 器 的 注册 动作 中 进行 处 理 ， 代 码 如 下 : 


例 程 19 代码 位 置 : 光盘 \TM\11known\application\modules\default\controllers\UserController.php 
/如果 用 户 通 过 POST 方法 提交 表单 
/获取 当前 控制 器 对 象 的 POST 方法 
SarrayProvice = Plugin Util ProvinceAndCityFactory::getProvinceO: 
S$p = $arrayProvice[$request->getParam('p")]: 

SarrayCity=Plugin_Util ProvinceAndCityFactory::getCityByProvinceArrayIndex($request->getParam('p")): 
Sc = $arrayCity[$request->getParam('c")]: 


if ($this->_request->isPost()) { 
Srequest = $this->_request: 


Sconfig = Zend Registry::get(config): 


1/ 获取 Zend_Config 对 象 
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SarraySmtpConfig = amray(auth => ‘login' . ‘usemame'’ => S$config[mail][username'] . password => 
S$config[mail][password]): /邮件 配置 信息 
S$transport = new Zend Mail Transport_ Smtp(Sconfig[mail][host]. SaraySmtpConfig): /构建 邮件 传输 对 象 


@ Smail = new Zend Mail(utf-8): /构建 Zend Mail 对 象 ， 并 设置 字符 编码 
S$mail->setSubject($config[mail']['subject]): /邮件 主题 
S$mail->setBodyHtml(file_get_contents(Sconfig[mail][bodyPath ])): /HTML 邮 件 内 容 
S$mail->setFrom($config[mail]['fom'], Sconfig[mail][name]): /发 件 人 
Smail->addTo(trim($request->getParam('‘email’))): 1// 收 件 人 

{ 
S$mail->send($transport): // 发 送 youjian 
} catch (Zend Exception $e) { 
$e->getMessage(): 
} 
// 保 存 用 户 信息 到 数据 库 
SarrayUserInfo = aray(netname’ => tim(Srequest->getParam(netname)) , password ”一 
md5(trim($request->getParam('password'))) "email 一 frim(Srequest->getParam(email)) "tel 一 
trim($request->getParam(tel)) , 'pc' => $p . ~’ . $c , ‘regtime' => date('Y-m-d His) , face' => " , ‘pubtimes' => 0 ,Teplytimes' 
=> 0 , 'score' => 20 , ‘usertype' => 0): /用 户 注册 信息 所 构建 的 数组 
Sthis->_userModel->insert($arrayUserInfo): // 保 存 用 户 注册 信息 
} catch (Zend Exception $e) { 
$e->getMessage(); 
} 
Sthis->_sessionNamespace->netname = trim(Srequest->getParam(netname)): 
Sthis->_redirect('/user/register-success); // 定 向 到 用 户 注册 成 功 提示 页 面 
exitO; 
} 
< 代码 贴 十 


@ 为 Zend Mail 的 构造 方法 指定 UTF-8 参数 的 作用 是 设 定 邮件 编码 为 UTF-8。 


上 述 代 码 保存 用 户 注册 信息 ， 主 要 通过 用 户 表 模型 Model DbTable_User 的 insert() 方 法 实现 , 该 方 
法 的 语法 格式 如 下 : 
insert(array insertArray) 


该 方法 中 主要 包含 一 个 关联 数组 参数 ， 该 数组 的 键 名 为 用 户 表 的 字段 名 ， 键 值 为 用 户 录入 的 注册 
信息 。 

为 了 提升 网 站 形象 ， 提 高 用 户 回访 数量 ， 在 保存 用 户 注册 信息 前 ， 一 般 需 要 向 注册 用 户 所 填写 的 
邮箱 中 发 送 一 封 注册 成 功 提示 邮件 。 发 送 邮 件 功 能 是 通过 Zend_Mail 实现 的 ， 使 用 Zend_Mail 发 送 邮 
件 相 对 简单 ， 实 例 Zend_Mail 对 象 后 ， 只 需 通 过 调用 该 类 的 几 个 方法 ， 为 邮件 指定 主题 、 内 容 、 发 件 
人 和 收 件 人 等 信息 , 最 后 再 通过 send0 方 法 将 邮件 发 送出 去 , Zend_Mail 类 中 常用 的 方法 如 表 11.8 所 示 。 


表 11.8 Zend_Mail 类 中 常用 方法 说 明 


属 性 值 说 了 明 
setSubjectO | 设置 邮件 主题 


setBody0) 设置 文本 格式 的 邮件 内 容 
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续 表 
属 性 什 说 明 
setBodyHtmlO | 设置 HTML 格式 的 邮件 内 容 
setFromO | 设置 发 件 人 地 址 
addTol 设置 收 件 人 地 址 
11.9.4 ”用 户 登录 的 实现 过 程 
用 户 注 册 成 功 后 访问 本 站 就 可 以 直接 使 用 已 注册 的 账号 登录 。“ 明 日 知道 ”网 站 的 用 户 登录 页 面 
如 图 11.31 所 示 。 


慎 9908 | 


全 三 三 
PY i | = 
a 


J] 沁 枉 双 忆 尝 家 上 | 


(中 ”1999-2009 吉 林 省 明日 科技 有 限 公司 


图 11.31 用 户 登录 页 面 


用 户 登 录 的 实现 步骤 如 下 : 
建立 用 户 登录 表单 。“ 明 日 知道 ”网 站 的 登录 表单 使 用 Zend_Form 制作 ， 代 码 如 下 : 
例 程 20 ”代码 位 置 : 光盘 \TM\11\known\application\forms\User\login.php 


Ch) 


class Form User Login extends Zend Form 


public function _ construct ($options = null) 
{ 
parent::_ construct($options): // 调 用 父 类 构造 函数 
Sthis->setName('form login') // 表 单 名 称 
->setMethod(post) /表单 提交 方法 
->setAction($options['baseUrl'] . /user/login') // 表 单 提交 地 址 
->addAttribs(array( 
"Style => margin:0px: padding:0px" /表单 样式 
)): 
© Sthis->addElements(array( // 为 表单 增加 用 户 昵称 录入 文本 框 


// 定 义 用 户 登 录 表 单 ， 继 承 自 Zend_Form 
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new Zend Form Element Text(netname', array( 


‘Tequired' => true, // 指 定 该 表单 元 素 是 必要 的 
"abel 一 昵称 : // 元 素 标签 
"attribs' => array( // 元 素 属性 


‘class' => 'input login form' 

'style' => 'position:absolute; left:32px: top:0px: width:180px: height:18px: line-height: 18px’ 
), 
‘filters' => array('StringTrim'), // 过 滤 掉 首尾 空格 
walidators => array( 

array(NotEmpty', true, array(messages' => '。 请 输入 登录 昵称 )) /用 户 昵称 非 空 验证 


» 
"decorators' => array( // 元 素 装 饰 器 
ViewHelper、 
aay(HtmlTag' array(tag=>'dd)), 
arTay(Label', array('tag=>'dt’)) 
) 
2 Zend Form Element Password('password', array( /为 表单 增加 用 户 密码 录入 文本 框 
Yequired' => true, // 指 定 该 表单 元 素 是 必要 的 
"abel => 密码 : // 元 素 标签 
"attribs' => array( // 密 码 框 属性 


‘class' => "input_login_form', 
'style' => 'position:absolute; left:32px: top:O0px:; width:180px: height:18px: line-height:18px; 


), 

‘filters' => array('StringTrim'), // 过 滤 掉 首尾 空格 

‘validators' => array( // 登 录 密码 非 空 验证 
array(NotEmpty', true, array(messages' => '。 请 输入 登录 密码 7) 

小， 

"decorators' => array( // 元 素 装 饰 器 
ViewHelper、 
array(HtmlTag', array('tag’ => 'dd)). 
array('Label', aray('tag’ => "dt)) 

) 
), 
new Plugin Form Element Vcode('vcode', array( // 验 证 码 

Tequired => true, 1/ 指定 该 表单 元 素 是 必要 的 

"label => 验证 码 : // 元 素 标签 

"attribs' => array( // 验 证 码 属性 
‘textClass' => ‘input_login_form'. 
‘textStyle' =>'position:absolute: left:32px: top:0px: width:60px: height:18px: line-height: 18px; ', 
"imageStyle' =>'position:absolute; left:100px: top:0px:;', 
"spanStyle' => 'position:absolute: left:42px: top:37px; height:18px', 
'aClass' => 'a4'. 
‘functionName' => 'changeValidateCode()' 

) 

‘filters' => array('StringTrim'). // 过 滤 掉 首尾 空格 


walidators' => array( 
amay(NotEmpty'. true. array(messages' =>'。 请 输入 验证 码 ")). /验证 码 非 空 验证 
new Plugin Validate VcodeRightO /验证 码 正确 性 验证 
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"decorators' => array( 
"ViewHelper'. 
array('HtmlTag'. array('tag’ => 'dd', 'style' => ‘height:55px:')). 
array(Label'. armray(tag 一 dt) 

) 


// 元 素 装 饰 器 


)， 

new Zend Form Element Image('submitImage', array( /提交 图 片 按钮 
Tequired => false, /该 元 素 值 可 为 空 
"abel =>", 


'src' => $options['baseUrl'] . /img/btn login.gif, /图 片 地 址 
"attribs' => array( 


‘style' => 'position:absolute; left:30px: top:Opx:" 
六 
} 
Ah) 代码 贴 十 
@ addElements0 方 法 用 于 将 表单 元 素 对 象 添 加 到 表单 中 。 


使 用 Zend_Form 制作 表单 ， 除 可 构建 表单 UI 界面 外 ， 还 可 以 同时 实现 用 户 录入 内 容 的 过 滤 、 验 
证 等 操作 ， 从 而 可 以 很 大 程度 地 提高 程序 开发 效率 。 使 用 Zend_Form 构建 表单 组 件 的 构造 方法 一 般 包 
含 两 个 参数 ， 分 别 为 表单 元 素 名 称 和 指定 该 元 素 属性 的 关联 数组 ， 该 数组 常用 的 键 值 如 表 11.9 所 示 。 


表 11.9 Zend_Form 表单 元 素 常用 键 值 说 明 


属 性 值 说 明 
required 用 于 指定 表单 元 素 值 是 否 是 必要 的 
label 表单 元 素 标签 
attribs 表单 元 素 属性 
filters 对 表单 元 素 进行 过 滤 
validators 对 表单 元 素 进行 效 验 
decorators 对 表单 元 素 UI 进行 装饰 


11.10 发 表 问 题 模块 设计 


当 用 户 在 “明日 知道 ”网 站 搜索 问题 时 ， 如 果 没 有 查找 到 预期 结果 ， 可 以 将 要 解决 的 问题 发 表 到 
该 站 ， 其 他 用 户 就 可 以 对 问题 进行 解答 或 探讨 。 


11.10.1 发 表 问 题 模块 概述 


如 果 用 户 已 经 注册 为 本 站 会 员 ， 并 成 功 登 录 本 站 ， 则 可 以 通过 “明日 知道 ”网 站 的 发 表 问题 模块 
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将 问题 发 表 出 去 ， 其 他 用 户 就 可 以 回答 该 问题 ， 最 终 可 以 帮助 提问 用 户 获得 问题 的 最 佳 答案 。“ 明 日 
知道 ”网 站 的 发 表 问题 表单 主要 包括 问题 主题 、 问 题 类 别 、 问 题 的 内 容 描 述 、 附 件 、 验 证 码 等 表单 元 


素 ， 如 图 11.32 所 示 。 


办 和 mS | 本 到 二 lB ZU wjee|ss ls lime dsme de | 


请 在 上 壕 编辑 句 中 簿 入 要 发 表 的 内 容 
峙 件 : 


请 格 图 中 所 星 示 的 内 容 输 入 到 上 述 误 吾 椎 中 


EE 
图 11.32 发 表 问 题 表 单 


11.10.2 ”发 表 问 题 模块 技术 分 析 


发 表 问题 模块 中 ， 应 该 掌握 的 主要 技术 包括 在 线 编辑 器 的 制作 、 图 片 验证 码 的 制作 、 文 件 上 传 的 
实现 过 程 。 其 中 ， 在 线 编辑 器 及 图 片 验证 码 的 制作 过 程 已 经 在 前 面 章节 中 进行 了 重点 讲解 ， 下 面 通过 
本 模块 介绍 利用 Zend Framework 实现 文件 上 传 的 过 程 。 

(1) 用 户 发 表 问题 表单 是 通过 Zend_Form 实现 的 ， 所 以 需要 使 用 Zend_Form Element File 类 建 
立 文 件 上 传 域 ， 代 码 如 下 所 示 : 

例 程 21 “代码 位 置 :光盘 \TM\11\known\application\forms\Bbs\Pubtitle.php 


new Zend Form Element File('filel' array( // 文 件 选 择 域 
‘required' => false. /指定 文件 上 传 域 并 非 必要 
"abel => 附件: /标签 
‘maxFileSize' => '30000000', /上 传 文件 大 小 


> 
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"description' => ' 请 选择 要 上 传 的 附件 /描述 
"attribs' => array( // 文 件 上 传 域 属性 
size => "60', 


‘class' => "input_pubtitle form' 


) 


a 一 array( // 非 空 验证 
atray(NotEmpty', false, array(messages' > ' 请 输入 内 容 )) 

y 

"decorators' => array( // 装 饰 器 
‘File', 


array('Description', array(tag 一 'dt')), 
amay(HtmlTag' array(tag' => 'dd) )， 
array(Label', array(tag => 'dt)) 

) 

), 

在 讲解 用 户 登 录 表单 时 ， 已 经 详细 介绍 了 Zend_Form 表单 及 表单 元 素 的 常用 属性 的 含义 。 上 述 代 
码 中 ， 使 用 maxFileSize' => 30000000' 指 定 所 允许 上 传 附件 的 最 大 值 。 运 行 发 表 问 题 模块 ， 上 述 代码 将 
转换 为 如 下 的 HTML 代码 : 

<input type="hidden" name="MAX FILE SIZE" value="30000000" id="MAX FILE SIZE"> 

<input type="file" name="filel" id="filel" size="60" class="input_pubtitle form"> 

(2) 当 用 户 提交 发 表 问 题 表单 后 ， 提 交 的 内 容 将 被 发 送 到 QuestionController 控制 器 的 pubtitleAction 
动作 进行 处 理 ， 其 中 用 户 实现 文件 上 传 的 关键 代码 如 下 : 


例 程 22 ”代码 位 置 ， 光盘 \TM\11\known\application\modules\default\controllers\QuestionController.php 


Sadapter = new Zend File Transfer Adapter_Http(): // 构 建 上 传 适配器 
SupfileName ="; // 上 传 后 ， 保 存在 服务 器 中 的 文件 名 
if ($adapter->getFileName('file1’) != null) { // 如 果 用 户 已 经 选择 了 上 传 文件 
SupfileDir = $this-> config[bbs']['upfilesdir]: // 从 配置 文件 中 读 取 上 传 文件 保存 目录 
if(!is_dir($upfileDir)) { // 判 断 上 传 目录 是 否 存 在 ， 如 果 不 存 在 则 创建 该 目录 
mkdir($upfileDir): 
} 
S$adapter->setDestination($upfileDir): // 为 上 传 适配器 设置 上 传 目 录 


S$arrayOldFileName = array_reverse(explode('.'. basename($adapter->getFileName('file1")))): 
/提取 上 传 文件 原来 名 字 
SextendsFileName ='" . SarrayOldFileName[0]: // 获 取 上 传 文件 扩展 名 
S$upfileName = date(YmdHis') . mt_rand(1000, 9999) . $extendsFileName: // 文 上 传 文件 指定 新 名 称 , 用 时 将 和 一 
个 4 位 随机 数组 成 
SupfilePathAndName = SupfileDir . " . SupfileName: /文件 在 服务 器 中 保存 目录 及 名 称 
S$adapter->addFilter(Rename' array('target’ => S$upfilePathAndName , 'overwrite' => true), 'filel1): /更 改 上 传 


文件 名 
Sadapter->addValidator('Size'. false. array(min' => 'OkB' . ‘max' => '2MB' . "bytestring' => false . Imessages' => '。 您 
所 上 传 的 文件 不 能 超过 2MB)): // 指 定 上 传 文件 的 范围 
S$adapter->addValidator(ExcludeExtension' false, array(php' . 'exe' , ‘messages' => '。 您 上 传 的 文件 类 型 不 允许 7): 
/指定 上 传 文件 扩展 名 
| 
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上 述 代码 中 是 通过 Zend File 实现 文件 上 传 的 ， 由 于 本 模块 上 传 采 用 HITP 方式 , 所 以 实现 文件 上 
传 主要 通过 Zend File Transfer Adapter Http 适配器 实现 ， 该 适配器 的 常用 方法 如 表 11.10 所 示 。 


表 11.10 Zend_File_ Transfer_Adapter_Http 适配器 常用 方法 说 明 


属 性 什 说 明 
getFileName0) | 获取 上 传 文件 的 名 称 
setDestination0) | 指定 上 传 文件 保存 到 服务 器 中 的 目录 
addFilter() | 为 上 传 增加 过 滤 规 则 
addValidatorO) 为 上 传 增加 效 验 规则 


11.10.3 发表 问题 模块 实现 过 程 


制作 “明日 知道 ”网 站 发 表 问题 模块 ， 主 要 包括 发 表 问题 表单 视图 的 建立 和 保存 问题 内 容 两 个 关 
键 环节 。 

(1) “明日 知道 ”网 站 发 表 问题 表单 使 用 Zend_Form 实现 ， 通 过 Zend_Form 实现 表单 的 制作 可 
以 简化 对 提交 数据 的 过 滤 、 效 验 、UI 布局 的 操作 ， 从 而 能 够 提高 开发 效率 ， 便 于 开发 人 员 分 层 开发 及 
做 好 日 后 生产 环境 下 的 维护 工作 。 本 站 实现 发 表 问题 表单 的 代码 如 下 : 

例 程 23 ”代码 位 置 : 光盘 \TM\11\known\application\forms\Bbs\Pubtitle.php 

class Form Bbs_Pubtitle extends Zend Form 


public function _ constrct ($options = null) 


{ 


parent::_construct(null): // 调 用 父 类 构造 方法 

Sthis->setName('form pubtitle) // 表 单 名 称 
->setMethod(post) /表单 提交 方法 
->setAction(Soptions['"baseUrl].Vquestion/pubtitle) /表单 提交 地 址 
->addAttribs(array( /表单 属性 


'enctype' => "multipart/form-data'. 
'style' => ‘margin:O0px: padding:Opx;', 
站: 
Sthis->addElements(array( 
new Zend Form Element Text('title', array( 
a // 省 略 代 码 用 于 设置 问题 主题 元 素 的 属性 
), 
Dew Zend Form Element _ Select(bbstypeid', array( 
汪 /省 略 代码 用 于 设置 问题 类 别 下 拉 列 表 属 性 
). 
new Plugin Form Element Editor(content. array( 
本 /省 略 代码 用 于 设置 问题 内 容 编辑 器 的 属性 
), 
new Zend Form Flement File('filel'. array( // 文 件 选 择 域 
A /省 略 代码 用 于 设置 文件 选择 框 的 属性 
). 
new Plugin Form Element Vcode(Cvcode' array( 
/省 略 代码 用 于 设置 验证 码 的 属性 
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). 
new Zend Form Element Button(submitButton'. array( 
5 /省 略 代码 用 于 设置 提交 按钮 的 属性 


new Zend Form Element Hidden(titleid', array( 
// 在 编辑 问题 时 ， 该 隐藏 域 用 于 保存 问题 D 的 值 
walue' => $options['titleid"] 
) 
): 


} 


上 述 Form_Bbs_Pubtitle 类 继承 自 Zend_Form 类 ， 这 样 该 类 就 可 以 使 用 Zend_Form 类 的 所 有 方法 ， 
如 使 用 setName0 方 法 设置 表单 名 称 、setMethod0 方 法 设置 表单 提交 方法 、setAction0 方 法 设置 表单 提 
交 地 址 等 ， 其 中 Zend_Form 表单 元 素 的 常用 类 如 表 11.11 所 示 。 


表 11.11 Zend_Form 表单 元 素 常用 类 说 明 


属 性 值 说 阴 
Zend Form Element_ Text 生成 文本 框 
Zend Form Element Select 生成 下 拉 列 表 框 
Zend Form Element File 生成 文件 选择 域 
Zend_ Form Element Button 生成 按钮 
Zend Form Element_Hidden 生成 隐藏 域 
Zend Form Element Checkbox 生成 复 选 框 
Zend_ Form Flement Image 生成 图 片 按钮 
Zend Form Element Password 生成 密码 框 
Zend Form Element Radio 生成 单 选 按 钮 
Zend Form Element Reset 生成 重 置 按钮 
Zend Form Element_ Submit 生成 提交 按钮 
Zend Form Element Textarea 生成 文本 域 
Zend Form Element Text 生成 文本 框 


(2) 用 户 提交 表单 后 ， 录 入 的 内 容 将 被 提交 到 QuestionController 控制 器 的 pubtitleAction 动作 被 
保存 ， 保 存 用 户 提交 内 容 的 核心 代码 如 下 所 示 : 


例 程 24 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\controllers\QuestionController.php 

SpubUser = $this->_userModel->findByNetname($this-> auth->getIdentity0): /获取 发 帖 用户 信 息 

$arrayInsert = array(title’ => $formData['title] . 'content => $formData['content] . "unhtmlcontent => 
$this->view->unHtml($formData['content]) , 
'addtime' => date(Y-m-d H:i:s') . 'istop' => false . user id' => $pubUser['id] . 'bbstype_id' => $formData['bbstypeid'] . "browse' 
=>0. filename' => 


SupfileName . 'isjh' => false): // 发 帖 内 容 数组 
Sthis-> titleModel->insert($arrayInsert): // 使 用 问题 模型 的 insert0 方 法 ， 将 问题 
信息 保存 到 数据 库 
SlastTitle = $this-> titleModel->findLastByUserid(SpubUser['id]): // 查 找 给 用 户 最 后 发 帖 信息 ， 即 刚 插入 


S3y 
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帖子 的 信息 
S$this->_redirect('question/thread/param/" . $lastTitle['id"]): // 使 页 面 定位 到 该 帖 的 详细 信息 页 


上 述 代码 首先 通过 用 户 模型 的 fndByNetname() 方 法 获得 当前 登录 用 户 的 信息 ， 然 后 将 问题 信息 保 
存 到 数据 库 的 tb title 表 中 ， 成 功 保存 问题 信息 后 ， 将 页 面 定向 到 所 发 问题 的 详细 信息 页 。 


11.11 回复 问题 模块 设计 
用 户 成 功 注 册 为 “明日 知道 ”网 站 会 员 后 ， 不 仅 可 以 发 表 问 题 ， 还 可 以 回复 问题 ， 与 其 他 用 户 进 
行 讨 论 。 
11.11.1 回复 问题 模块 概述 


“明日 知道 ”网 站 的 问题 回复 表单 利用 Zend_Form 制作 ， 主 要 包括 回复 主题 、 回 复 内 容 和 验证 码 
等 表单 元 素 ， 如 图 11.33 所 示 。 


回复 主题 , 


请 在 上 述 文本 框 中 输入 四 和 揽 主题 


回复 内 容 ， 


请 在 上 述 编 回 器 中 输入 要 辐 复 的 内 容 


请 格 图 中 所 显示 的 内 宕 绩 入 到 上 述 文本 答 中 
BE 
图 11.33 ”问题 回复 表单 


11.11.2 回复 问题 模块 技术 分 析 


和 


回复 问题 模块 中 应 该 掌握 的 主要 技术 包括 在 线 编辑 器 的 制作 和 使 用 方法 、 图 片 验证 码 的 制作 方法 ， 
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以 及 问题 引用 的 实现 。 其 中 在 线 编辑 器 和 图 片 验证 的 实现 方法 在 上 面 已 经 介绍 ， 下 面 介绍 “明日 知道 ” 
网 站 问题 引用 的 实现 过 程 ， 其 关键 代码 如 下 : 

$value ='<FIELDSET><LEGEND> 引 自 : 楼 主 <LEGEND> 引 用 的 内 容 </FIELDSET>'; 

将 上 述 代码 作为 编辑 器 的 value 值 ， 将 在 编辑 器 显示 如 图 11.34 所 示 的 效果 。 通 过 上 述 代码 可 知 ， 
引用 可 以 通过 HTML 标签 的 <FIELDSET> 标 签 实 现 ，<FILELDSET> 标 签 用 于 在 文本 或 其 他 元 素 外 绘制 
一 个 边框 ， 如 果 在 该 标签 内 使 用 <LEGEND> 子 标签 指定 标题 ， 则 在 边框 左上 侧 还 会 显示 指定 的 标题 。 


引 自 : 楼主 


我 是 今年 刚 工作 的 程序 员 ， 对 一 些 代码 的 运用 还 不 是 很 熟悉 ， 对 代码 的 书写 还 是 不 很 规范 。 公 司 要 我 开 
发 一 个 项 目 ， 我 有 点 摸 不 着 头绪 ， 朋 友 买 的 编程 词典 我 看 了 一 下 ， 就 得 很 实用 ， 里 面 的 代码 是 编辑 好 

的 ， 直 接 就 可 以 实用 ， 省 了 我 不 少 的 时 间 啊 ， 经 理 还 说 我 做 的 快 ， 给 我 发 了 奖金 ， 有 机 会 一 定 要 请 你 们 
吃饭 呢 


图 11.34 问题 引用 
11.11.3 ”回复 问题 模块 实现 过 程 


问题 回复 模块 ， 主 要 包括 回复 表单 的 设计 和 保存 回复 信息 两 个 主要 过 程 。 

(1) 同样 ， “明日 知道 ”网 站 的 问题 回复 表单 采用 Zend_Form 制作 ， 其 关键 代码 如 下 : 
例 程 25 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\controllers\QuestionController.php 
class Form_ Bbs_Reply extends Zend Form 


public function _ construct ($options = null) 


{ 
parent::_ construct(null); // 调 用 父 类 构造 函数 


Sthis->setName('form reply) /设置 回复 表单 名 称 
->setMethod(post) /设置 表单 提交 方法 
->SsetAction($options["baseUrl]./question/thread/param/ . Soptions['titleid] .'-1-T) 
/设置 表单 提交 地 址 
->addAttribs(aray( /1/ 设 置 提交 表单 属性 
'style' => 'margin:O0px: padding:Opx:", 
六 
Sthis->addElements(array( 
new Zend Form Element Text('title'. array( 
ee // 省 略 代码 用 于 设置 回复 主题 文本 框 属性 
). 
new Plugin Form Element Editor('content'. array( 
/省 下 代码 用 于 设置 编 部 吕 必 性 
7 
new Plugin Form Element Vcode(vcode' array( 
本 这 /省 略 代码 用 于 设置 验证 码 属性 


new Zend Form Element Button(submitButton', array( 
// 省 略 代 码 用 于 设置 提交 按钮 属性 
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上 述 代码 中 的 Form Bbs Reply 类 继承 自 Zend_Form 类 ， 通 过 Zend Form 类 中 定义 的 方法 来 为 回 
复 问题 表单 设置 属性 ， 并 通过 addElements() 方 法 将 表单 元 素 对 象 添 加 到 问题 回复 表单 中 。 

(2) 用 户 输入 完成 回复 内 容 , 提交 表单 后 ,所 有 回复 内 容 将 被 提交 到 QuestionController 控制 器 的 
threadAction 动作 中 进行 保存 ， 其 中 用 于 保存 用 户 回复 信息 的 关键 代码 如 下 


例 程 26 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\controllers\QuestionController.php 


if ($this-> request->isPost()) { // 判 断 用 户 是 否 提 交 了 表单 
SformData = $this-> request->getPost(); // 获 取 表单 提交 数据 所 组 成 的 数组 
if (SreplyForm->isValid($formData)) { // 如 果 通 过 表单 验证 
Sreplyid = $this-> request->getParam(replyid"): 。“”// 获 得 问题 类 别 ID 
if (isset(Sreplyid) && Sreplyid != null) { // 如 果 设 置 了 问题 类 别 [D， 则 说 明 要 进行 更 改 问题 回复 


操作 ， 和 否则 进行 添加 问题 回复 操作 
$updateArray = array('title' => $formData['title] , "content => $fiormData['content]): 


// 更 改 问题 回复 数组 
Swhere = $this-> replyModel->getAdapter()->quoteInto('id = ?'", Sreplyid); // 更 改 条 件 
Sthis->_replyModel->update($updateArray, Swhere): // 更 改 数据 


Sthis->_redirect('/question/thread/param/" . $titleid . '-' . Sthis-> request->getParam(topage) . '#r' . $replyid); 

// 重 定向 到 问题 详细 信息 页 面 
}else{ 

SreplyUser = $this->_userModel->findByNetname($this-> auth->getIdentity0); /获取 回复 人 信息 

SinsertArray = array(title’ => $formDataf'title] ，'content => $formData['content] , ‘addtime' 一 
date('Y-m-d H:is) , user id 一 SreplyUser[fid] , ‘title id => SarrayTitle["tid'] . 'topindex' => 0 , 'bbstype id 一 
SarrayTitle['bbstype_id"]): // 添 加 问题 回复 数组 

$this-> replyModel->insert(SinsertArray): // 添 加 数据 

StotalReply = $arrayTitle['totalreply’] + 1: 


0 StotalPage = ceil($totalReply / $pageSize); 
@ Sthis->_redirect(/question/thread/param/' . $titleid . '-'. StotalPage . #b"):”// 重 定向 到 问题 详细 信息 页 面 
} 
} else{ 
foreach ($replyForm->getMessages() as SmessageArray) { // 获 取 错 误 信 息 
foreach (SmessageArray as Smessage) { 
SerrorMsg .= $message . '<br />": 
1 
} 
SreplyForm->populate($formData): // 表 单数 据 回填 
; 
} 
< 代 码 贴 二 


@ 函数 ceil0 的 作用 是 对 浮 点 数 进行 向 上 取 整 。 
@_redirect0 方 法 的 作用 是 使 当前 页 面 重 定向 到 其 他 页 面 。 


上 述 代 码 首先 通过 当前 控制 器 对 象 的 _request 属性 的 isPost0 方 法 判断 用 户 是 否 提交 表单 ， 如 果 是 
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则 首先 对 表单 提交 的 数据 进行 数据 效 验 ， 如 果 成 功 通过 效 验 ， 则 通过 回复 表 〈tb_reply) 模型 的 msertO 
方法 将 回复 信息 保存 到 数据 库 中 。 


11.12 程序 调试 与 错误 处 理 


在 程序 开发 工程 中 ， 难 免 会 出 现 各 种 错误 ， 为 了 能 够 快速 地 解决 程序 开发 过 程 中 所 遇 到 的 错误 ， 
需要 开发 人 员 具 有 良好 的 编程 习惯 和 敏锐 的 观察 能 力 ， 并 不 断 地 培养 个 人 的 开发 经 验 。 本 节 将 对 程序 
开发 和 运行 过 程 中 常见 错误 的 调试 和 处 理 方法 进行 讲解 。 


11.12.1 项目 开发 过 程 中 常见 错误 及 解决 方法 


为 了 能 够 显示 项 目 开 发 过 程 中 的 错误 提示 ,需要 开发 php.ini 文件 的 display_startup_errors 和 display 
_errors 项 ， 或 者 在 本 站 的 application.ini 文件 中 加 入 如 下 配置 来 使 项 目 在 开发 过 程 中 显示 错误 提示 。 

phpSettings.display_startup_errors = 1 

phpSettings.display_errors = 1 

这 里 需要 注意 , 在 项 目 投入 生产 后 , 需要 将 上 述 两 项 的 值 设 为 0, 这 样 程序 在 运行 过 程 中 即使 出 现 
错误 ， 也 不 会 在 页 面 中 打印 出 错误 信息 ， 可 以 防止 网 站 的 漏洞 被 非法 用 户 发 现 ， 从 而 有 效 地 提高 了 网 
站 的 安全 性 。 开 启 上 述 错 误 提 示 信 息 后 ， 在 项 目 开 发 过 程 中 一 旦 遇 到 错误 ， 开 发 人 员 可 以 根据 错误 提 
示 推 断 错误 所 在 ， 并 予以 解决 。 

在 PHPS 以 上 版 本 中 ， 引 入 了 异常 处 理 机 制 ， 开 发 人 员 通 过 异常 处 理 可 以 捕获 程序 运行 时 的 错误 。 
异常 处 理 中 主要 包括 tty、catch 和 throw 这 3 个 关键 字 ， 其 语法 结构 如 下 所 示 : 


ty{ 

// 要 执行 的 代码 块 
}catch(Exception $e){ 

// 异 常 处 理 方式 ， 这 里 可 以 打印 错误 信息 或 使 用 throw 语 句 抛 出 异常 
] 


用 户 要 执行 的 代码 块 编写 在 try/catch 代码 块 之 间 ， 通 过 catch 语句 指定 要 捕获 的 异常 ， 然 后 通过 
throw 语句 抛 出 异常 。 


11.12.2 自 定义 错误 页 面 


为 了 提高 页 面 的 友好 程度 ， 隐 藏 项 目 生产 环境 下 的 错误 ， 往 往 需 要 在 项 目 中 制作 一 个 单独 的 用 于 
显示 错误 提示 的 页 面 。“ 明 日 知道 ”网 站 的 错误 提示 页 面 如 图 11.35 所 示 。 

使 用 Zend Framework 框架 制作 项 目 时 ， 有 一 个 专门 的 错误 控制 器 ErrorController 用 来 显示 页 面 错 
误 提 示 ， 开 发 人 员 只 需 在 该 控制 器 中 定义 一 个 名 为 errorAction 的 动作 即 可 。“ 明 日 知道 ”网 站 中 的 错 
误 控 制 器 代码 如 下 : 
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个 对 不 起 ， 您 要 浏览 的 页 面 不 存在 或 发 生 未 知 错误 ， 请 与 本 站 管理 员 联 系 | 


人 加 ”1999.2010 吉 林 省 明日 科技 有 限 公司 


图 11.35 自 定义 的 错误 提示 页 面 


例 程 27 ”代码 位 置 : 光盘 \TM\11\known\application\modules\default\controllers\QuestionController.php 
class ErrorController extends Zend Controller Action 


public function errorAction 0 
Sthis->view->title = "明日 知道 网 站 错误 提示 ": /页 面 标题 
$this->_ helper->layout->disableLayoutO: // 去 掉 布局 
} 
} 


定义 完成 上 述 错误 控制 器 后 ， 还 需要 定义 errro.phtml 视图 ， 该 视图 用 来 显示 错误 提示 的 内 容 。 
11.13 本 章 总 结 


“明日 知道 ”网 站 力求 打造 成 为 国内 知名 的 专业 软件 开发 技术 搜索 和 交流 平台 。 通 过 该 平台 开发 
人 员 和 编程 初学 者 可 以 查找 到 大 部 分 开发 和 学 习 过 程 中 可 能 会 遇 到 的 技术 难题 ， 即 使 没有 查找 到 预期 
结果 ， 还 可 以 通过 该 平台 将 问题 发 表 出 去 ， 这 样 就 可 以 与 其 他 网 友 探讨 ， 最 终 将 问题 解决 。 

网 站 开发 人 员 在 开发 过 程 中 ， 通 过 严谨 的 需求 分 析 、 功 能 分 析 和 数据 库 分 析 的 全 过 程 ， 并 充分 考 
虑 到 生产 环境 下 的 升级 与 维护 等 后 期 工作 , 采用 Zend 公司 研发 的 MVC 框架 Zend Framework 实现 。 在 
开发 过 程 中 ， 使 用 到 目前 Web 开发 领域 常用 的 热门 技术 ， 如 在 线 编辑 器 、 图 片 验证 码 、 类 似 Google 
搜索 条 的 下 拉 关 键 字 提示 列表 、 关 键 字 分 词 查询 和 查询 结果 中 关键 字 描 红 输 出 等 。 

通过 学 习 本 章 内 容 ， 可 以 对 项 目的 完整 开发 流程 有 所 了 解 ， 并 理解 MVC 开发 模式 的 含义 。 在 技 
术 方 面 ,不 仅 可 以 掌握 目前 Web 应 用 的 主流 模块 和 功能 的 制作 方法 , 而 且 可 以 熟练 应 用 Zend Framework 
框架 进行 项 目 开发 。 如 果 读 者 能 够 潜心 学 习 本 章 内 容 ， 相 信 定 能 受益 匪 浅 ， 使 个 人 技术 水 平 有 显著 的 
提高 。 


中 
ER 
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电子 商务 网 站 


(Apache+PHP+ADODB+Smarty+Ajax+MySQL 实现 ) 
(器 " 视频 讲解 : 4 小 时 41 分 钟 ) 


随 着 20 世纪 PC 机 (个 人 计算 机 ) 的 发 展 和 互联 网 的 普及 ， 电 子 商务 从 报 文 
时 代 进 入 到 了 Internet 时 代 ， 并 逐渐 为 大 众 所 了 解 和 接受 。 电 子 商 务 (Electronic 
Commerce， 简 称 EC) 是 目前 发 展 较 快 的 一 种 商务 模式 。 迄 今 为 止 ,不同 领域 的 人 
对 EC 的 理解 各 有 不 同 。 简 单 地 说 ，EC 是 一 种 基于 Internet， 利 用 计算 机 硬件 、 软 
件 等 现 有 设备 和 协议 进行 各 种 商务 活动 的 方式 。 

通过 阅读 本 章 ， 读 者 可 以 学 习 到 : 


了 解 如 何 进行 系统 分 析 

了 解数 据 库 设计 流程 

熟悉 搭建 系统 架构 的 方法 

掌握 Smarty 模板 技术 的 应 用 

掌握 ADODB 类 库 技 术 的 应 用 

熟悉 Smarty 模板 和 ADODB 类 库 的 配置 方法 
掌握 注册 即时 验证 的 实现 方法 

掌握 简单 的 树 形 菜单 的 实现 方法 
掌握 购物 车 的 实现 方法 

掌握 订单 的 处 理 方法 


豆 于 于 于 于 于 于 于 于 至 
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12.1 开发 背景 


自 20 世纪 90 年 代 以 来 ， 互 联网 的 蓬勃 发 展 ， 为 企业 提供 了 一 个 全 新 的 机 遇 ， 企 业 网 站 、 电 子 商 
务 成 为 热门 话题 。 其 中 ， 电 子 商务 更 是 关系 到 国家 的 经 济 结构 、 产 业 升 级 和 整体 经 济 竞争 力 。 为 此 ， 
我 国 已 经 将 发 展 电子 商务 列 为 信息 化 建设 的 重要 内 容 ， 并 努力 创造 条 件 ， 积 极地 推进 电子 商务 的 发 展 。 

据 美国 在 线 (AOL) 和 Henley Centre 联合 进行 的 一 项 调查 显示 ， 国 外 有 80% 的 受 调查 者 会 选择 网 
上 购物 或 寻求 帮助 ，10% 的 受 调查 者 会 选择 熟悉 的 品牌 或 厂商 来 购买 。 而 在 国内 ， 自 1997 年 拉 开 了 电 
子 商 务 的 序幕 ， 短 短 的 10 年 时 间 里 ， 全 国 已 有 4 万 家 商业 网 站 ， 几 乎 每 天 都 有 新 的 网 站 诞生 ， 厂 商 所 
在 地 也 从 北京 、 上 海 、 广 州 、 深 圳 等 东部 及 沿海 发 达 地 区 扩展 到 全 国 各 大 中 型 城市 。 


12.2 需求 分 析 


随 着 “地 球 村 ”概念 的 兴起 ， 网 络 已 经 深入 到 人 们 生活 的 每 一 个 角落 。 世 界 越 来 越 小 ， 信 息 的 传 
播 越 来 越 快 ， 内 容 也 越 来 越 丰富 。 现 在 ， 人 们 对 于 在 网 络 上 寻求 信息 和 服务 已 不 再 满足 于 简单 的 信息 
获取 ， 更 多 的 是 需要 在 网 上 实现 方便 、 便 捷 、 可 交互 式 的 网 络 服务 。 电 子 商 务 则 正好 满足 了 人 们 的 需 
求 。 它 可 以 让 人 们 在 网 上 实现 互动 地 交流 及 足 不 出 户 地 购买 产品 ， 向 企业 发 表 自 己 的 意见 、 服 务 需 求 
及 有 关 投 诉 ， 并 且 通 过 网 站 的 交互 式 操作 向 企业 进行 产品 的 咨询 ， 得 到 相应 的 回馈 及 技术 支持 。 精 明 
的 商家 绝 不 会 错过 这 样 庞大 的 市 场 ， 越 来 越 多 的 企业 已 经 开展 了 电子 商务 活动 。 加 入 电子 商务 的 行列 
也 许 不 会 让 企业 马上 见 到 效益 ， 但 不 加 入 则 一 定 会 被 时 代 所 抛弃 。 


12.3 系统 分 析 


12.3.1 系统 目标 


根据 客户 提供 的 需求 和 对 实际 情况 的 考察 与 分 析 ， 该 电子 商务 网 站 应 该 具备 如 下 特点 : 
首页 设计 要 能 够 吸引 用 户 的 目光 ， 整 个 页 面 要 以 简洁 为 主 ， 突 出 重点 。 

可 操作 性 强 ， 避 免 复杂 的 、 有 异议 的 链接 。 

浏览 速度 快 ， 尽 量 避 免 长 时 间 打 不 开 页 面 的 情况 发 生 。 

商品 信息 部 分 有 实物 图 例 ， 图 像 清楚 、 文 字 醒 目 。 

详细 的 商品 查询 功能 ， 可 以 通过 商品 的 各 个 属性 来 搜索 。 

详细 的 流程 介绍 ， 从 浏览 商品 到 购买 结账 ， 各 个 步骤 之 间 的 联系 最 好 能 以 图 例 来 说 明 。 
提供 在 线 咨询 。 

后 台 可 以 对 用 户 信息 和 商品 信息 进行 详尽 地 查看 和 管理 。 


办 办 罗 办 办 办 多 
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回 ”订单 管理 。 
易 维 护 ， 并 提供 二 次 开发 支持 。 


12.3.2 ”系统 功能 结构 


电子 商务 平台 分 为 前 台 系 统 和 后 台 系 统 。 下 面 分 别 给 出 前 、 后 台 的 系统 功能 结构 图 。 电 子 商务 前 
台 系 统 功能 结构 图 如 图 12.1 所 示 。 


电子 商务 前 台 系统 功能 结构 图 


灌 评 或 汗 
下 仿 薄型 
家 昌国 洲 


图 12.1 电子 商务 前 台 系 统 功能 结构 图 
电子 商务 后 台 系 统 功 能 结构 图 如 图 12.2 所 示 。 


电子 商务 后 台 系 统 功能 结构 图 


江 梧 训 


12.2 电子 商务 后 台 系统 功能 结构 图 


$39 
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12.3.3 ”功能 预览 


电子 商务 网 站 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 
购物 的 几 个 典型 功能 的 页 面 ， 其 他 页 面 可 参见 光盘 中 的 源 程序 。 
鹏 逊 购物 系统 主页 展示 网 站 的 部 分 最 新 商品 、 热 门 商品 、 推 荐 商品 以 及 网 站 的 最 新 公告 和 会 员 登 


下 面 列 出 鹏 逊 


录 窗 口 ， 如 图 12.3 所 示 。 最 新 商品 展示 页 面 分 页 展示 网 站 的 所 有 最 新 商品 ， 如 图 12.4 所 示 。 


图 12.3 网 站 主页 面 


图 12.4 最 新 商品 展示 页 面 
购物 车 页 面 展 示 会 员 在 本 站 购买 的 商品 ， 如 图 12.5 所 示 。 


商品 名称 购买 数量 市 场 价格 会 员 价格 扣 束 合计 
自行 车 [| 450 405 405 
永 RE 院 | 区 |] 4500 4050 9 alo0 
洗衣 机 提 ] 1000 00 a 2700 
[7 [| 四 1 日 
全 过 反 迁 。 出 辽 光 反 EE Ee 共计 :zm 元 


12.5 ”购物 车 页 面 


收银 台 页 面 完成 收 货 人 信息 的 填写 ， 并 提交 订单 ， 该 页 面 显示 游客 注册 时 需要 填写 的 昵称 、 密 码 
和 个 人 资料 等 相关 内 容 ， 如 图 12.6 所 示 。 


本 


| 


送 和 方式 : || 快递 下 


付 灶 方式 :| 支付 宇 “要 | 


12.6 ”收银 台 页 面 
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12.3.4 系统 流程 图 


为 了 便于 开发 人 员 了 解 系统 各 个 功能 模块 之 间 的 联系 及 完整 的 购物 流程 ， 下 面 给 出 系统 的 流程 图 ， 
如 图 12.7 所 示 。 


图 12.7 系统 流程 图 


12.3.5 ”开发 环境 


在 开发 电子 商务 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 : 
.服务 器 端 

操作 系统 : Windows 2003 Server/Linux (推荐 ) 。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.12.3。 
ADODB 类 库 。 


办 办 办 办 办 多 
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Smarty 模板 。 

开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效 果 为 1024X 768 像素 。 
客户 端 

浏览 器 : 推荐 正 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效 果 为 1024X768 像素 。 


回回 加 加 


因 轩 及 


12.3.6 ”文件 夹 组 织 结构 


用 户 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 例如， 创建 一 个 名 为 images 的 
文件 夹 ， 用 于 保存 程序 中 所 使 用 的 图 片 ) ， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 系统 的 整 
体 架构 。 因 为 本 项 目 使 用 的 是 Smarty+ADODB 技术 ， 所 以 目录 较 多 。 下 面 列 出 本 系统 的 目录 结构 〈 到 
三 级 目录 ) ， 如 图 12.8 所 示 。 


日 国 12\ShoppingSysten 


in 


电子 商城 系统 目录 


a 
Ds 


回 - 力 -图 -图 -图 


seemooooO0 


Smartyl 和 时 文件 


数据 库 连 接 、 管 理 、 分 页 类 和 Snarty 和 配置 类 的 实例 化 文件 
ine. php 一 一 Snarty 模 板 配 置 类 文件 
cess 样式 文件 


C=] 
田 -入 in 页 面 背景 图 片 文件 
田 - 留 ine 公共 资源 文件 
田 - 留 js javasceript 脚 本 文件 
田 - 国 pies 商品 图 片 文件 
日 岛 ysten AD0DB 类 库 、Snmarty 类 库 的 操作 文件 严 
田 - 国 sdodb adodb 类 库 
田 国 cache 一 一 一 一 一 一 一 Smarty 醒 存 文件 
日 筷 Snarty 配 置 文件 
日 入 Snarty 模 板 类 库 
日 仿 Saarty 必 时 文件 
全 5 ine. php- 数据 库 连 接 、 管 理 和 分 页 类 文件 
畏 s nm 数据 库 连 接 、 管 理 、 分 页 类 和 Smart 配置 类 的 实例 化 文件 
而 system. smarty. ine. php- smarty 弄 板 配置 类 文件 


图 12.8 ”电子 商务 网 站 文件 夹 组 织 结构 图 
12.4 ”数据 库 设计 


无 论 是 什么 系统 软件 ， 其 最 根本 的 功能 就 是 对 数据 的 操作 与 使 用 。 所 以 ， 一 定 要 先 做 好 数据 的 分 
析 、 设 计 与 实现 ， 然 后 再 实现 对 应 的 功能 模块 。 
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12.4.1 ”数据 库 分 析 


根据 需求 分 析 和 系统 的 功能 流程 图 , 找 出 需要 保存 的 信息 数据 (也 可 以 理解 为 现实 世界 中 的 实体 )， 
并 将 其 转化 为 原始 数据 (属性 类 型 ) 形式。 这 种 描述 现实 世界 的 概念 模型 ， 可 以 使 用 E-R 图 来 表示 ， 
也 就 是 实体 -联系 图 ， 最 后 将 E-R 图 转换 为 关系 数据 库 。 这 里 重点 介绍 几 个 E-R 图 。 

1. 会 员 信息 实体 

会 员 信息 实体 包括 编号 、 用 户 名 、 密 码 、E-mail、 身 份 证 号 、 联 系 电话 、QQ 号 、 密 码 提示 、 密 保 
答案 、 邮 编 、 注 册 时 间 、 真 实 姓名 等 属性 。 会 员 信息 实体 E-R 图 如 图 12.9 所 示 。 


图 12.9 会 员 信息 实体 E-R 图 


2. 商品 信息 实体 

商品 信息 实体 包括 编号 、 名 称 、 上 市 时 间 、 添 加 时 间 、 型 号 、 图 片 、 库 存 、 销 售 、 商 品类 型 、 会 
员 价 、 市 场 价 、 是 否 打 折 等 属性 。 商 品 信息 实体 E-R 图 如 图 12.10 所 示 。 

除 上 面 介绍 的 两 个 E-R 图 ， 还 有 商品 订单 实体 、 商 品评 价 实体 、 公 告 实体 、 管 理 员 实体 及 类 型 实 
体 和 友情 链接 实体 等 ， 限 于 篇 幅 ， 这 里 仅 列 出 主要 的 实体 E-R 图 。 


12.4.2 ”创建 数据 库 和 数据 表 


系统 E-R 图 设计 完成 后 ， 接 下 来 根据 E-R 图 来 创建 数据 库 和 数据 表 。 首 先 来 看 一 下 电子 商务 平台 
所 使 用 的 数据 表 的 情况 ， 如 图 12.11 所 示 。 
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图 12.10 商品 信息 实体 E-R 图 
四 服务 器 : localhost 》 怕 数据 库 : db_shop 
表 类 型 整理 


说 明 
tb_admin 。 MyISAM ”gb2312_chinese_cl 管理 员 信息 表 
tb_class MySAM gb2312.chinese_ci 商品 类 型 表 
rb_commo MySAM gb2312_chinese_cl 商品 信息 表 


tb_form MYISAM gb2312_chinese_ci 商品 订单 表 
tb_links MySAM gb2312_chinese_cl 友情 链接 表 
tb_opinion MySAM gb2312_chinese_cl 商品 评论 束 
tb_public MySAM ”gb2312_chinese_-cl 公 洁 信息 表 
tb_user MYISAM ”gb2312_chinese_cl 会 员 信息 表 


图 12.11 电子 商务 数据 表 
下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 
tb_admin 〈 管 理 员 信息 表 ) 
管理 员 信息 表 主要 用 于 存储 管理 员 的 信息 ， 其 结构 如 图 12.12 所 示 。 


胃 服务 器 : localhost 》 硬 数据 库 - db_shop ， 图 表 : tb_admin 


字段 类 型 整理 属性 Null 默认 额 外 说 明 
过 int(4) 否 auto_increment 。 自动 编号 

name varchar(50) gb2312_chinese_ci 否 管理 员 账号 
pwd varchar(20) gb2312_chinese_ci 否 管理 员 密 码 


图 12.12 ”管理 员 信息 表 结构 


tb_class 《商品 类 型 列表 )》 
商品 类 型 列表 主要 用 于 添加 商品 的 类 别 ， 可 以 设 定 多 个 子 类 别 〈 目 前 最 多 只 能 到 二 级 子 类 别 ) ， 
其 结构 如 图 12.13 所 示 。 
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图 服务 器 : localhost ， 大 数据库: db_shop ， 国 表 : tb_class 


字段 类 型 整理 属性 Null 黑 认 额外 说 明 
[rl ine(4) 否 auto_increment 自动 篇 号 
name varchar(20) gb2312_chinese_ci 否 类 型 名 称 
level int0) | 区 等 级 


图 12.13 ”商品 类 型 列表 结构 


tb_commo 〈 商 品 信息 表 ) 
商品 信息 表 主 要 用 于 存储 商品 的 相关 信息 ， 其 结构 如 图 12.14 所 示 。 


加 服务 器: localhost 》 怕 数据 库 - db_shop ， 图 表 : tb_commo 
字段 美 型 整理 


属 Null 黑 额外 说 明 

性 认 
uu int(4) 否 auto_increment ”自动 编号 
name varchar(50) gb2312-chinese_ci 否 商品 名 称 
pics Varchar(200) gb2312_chinese_ci 否 商品 图 片 
info mediumtext gb2312_chinese_ci 否 商品 介绍 
addtime dare 否 竹 )08 间 
area varchar(50) 。 gb2312_chinese_cl 否 商品 产地 
model varchar(50) gb2312_chinese_ci 否 商品 型 号 
class varchar(50) gb2312-chinese-ci 否 商品 类 别 
brand varchar(50) gb2312_chinese_ci 否 商品 品牌 
int(4) 王 机 | 所 商品 库存 
int(4) 否 0 商品 销量 
float 吾 市 场 价格 
float 否 会 员 价 格 
int(2) i 打折 这 
int0) | 是 否 新 品 
int0) 否 0 是 否 推荐 


图 12.14 商品 信息 表 结 构 


此 外 还 有 商品 订单 表 、 商 品 公告 表 、 用 户 信息 表 、 友 情 链接 表 和 商品 留言 表 ， 限 于 篇 幅 ， 这 里 不 
再 一 一 介绍 ， 相 关内 容 可 参见 本 书 附 赠 光 盘 中 的 数据 库 文件 。 


12.5 公共 文件 设计 


公共 模块 就 是 将 多 个 页 面 都 可 能 使 用 到 的 代码 写成 单独 的 文件 , 在 使 用 时 只 要 用 include 或 require 
语句 将 文件 包含 进来 即 可 。 如 本 系统 中 的 数据 库 连 接 、 管 理 和 分 页 类 文件 ，Smarty 模板 配置 类 文件 ， 
类 的 实例 化 文件 ，CSS 样式 表 文 件 ，js 脚本 文件 等 。 以 前 台 系 统 为 例 ， 下 面 给 出 主要 的 公共 文件 ， 后 
台 的 公共 文件 与 前 台大 同 小 异 。 


12.5.1 数据 库 连 接 、 管 理 和 分 页 类 文件 


在 数据 库 连 接 、 管 理 和 分 页 类 文件 中 定义 3 个 类 ， 分 别 为 ConDB 数据 库 连 接 类 ， 用 于 实现 通过 
ADODB 连接 MySQL 数据 库 ，AdminDB 数据 库 管 理 类 ， 通 过 使 用 ADODB 类 库 中 的 方法 执行 对 数据 
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库 中 数据 的 查询 、 添 加 、 更 新 和 删除 操作 ; SepPage 分 页 类 ， 用 于 对 商城 中 的 数据 进行 分 页 输出 。 实 现 


代码 如 下 所 示 : 


例 程 01 代码 位 置 : 光盘 \TM\12\ShoppingSystem\system\system class.inc.php 


<?php 
/其 据 库 连 接 类 
class ConnDB{ 
Var $dbtype: 
Var $host:; 
Var $user; 
Var Spwd; 
Var $dbname; 
Var Sdebug; 
Var $conn; 


function ConnDB($dbtype, $host, $user, S$pwd,$dbname, $debug=false){ 


$this->dbtype=$dbtype: 
S$this->host=$host: 

Sthis->user=$user; 

Sthis->pwd=$pwd: 


Sthis->dbname=$dbname; 
Sthis->debug=$debug; 

} 

function GetConnIdO{ 


require("adodb/adodb.inc.php"): 
if($this->dbtype—"mysql"){ 


S$this->conn=NewADOConnection("mysql"):; 
S$this->conn->Connect($this->host.$this->user. $this->pwd.$this->dbname); 


} 


S$this->conn->Execute("set names gb2312"); 


if($this->dbtype=——"mysql") 


Sthis->conn->debug=$this->debug: 


return S$this->conn; 


} 
function CloseConnIdO{ 
$this->conn->DisconnectO: 


} 


} 
/数据 库 管理 类 
class AdminDB{ 
function ExecSQL($sqlstr.$conn){ 
S$sqltype=strtolower(substr(trim($sqlstr).0.6)): 
Srs=$conn->Execute($sqlstr); 
if($sqltype—"select"){ 
Sarray=$rs->GetRows(); 
if(count($array)=—=0 || Srs—false) 
Tetum false: 
else 
Tetum $array: 


/构造 方法 ， 为 成 员 变量 赋值 


/实现 与 不 同 数据 库 的 连接 ， 并 返回 连接 对 象 
// 调 用 ADODB 类 库 文件 

// 判 断 成 员 变 量 传 递 的 数据 库 类 型 

// 执 行 与 MySQI 数 据 库 的 连接 

1/ 用户、 密码 


/设置 数据 库 的 编码 格式 


/返回 连接 对 象 


/定义 关闭 数据 库 的 方法 
/执行 关 闭 的 操作 


/定义 方法 ， 参 数 为 SQI 语 句 和 连接 数据 库 返 回 的 对 象 
/截取 SQL 中 的 前 6 个 字符 串 ， 并 转换 成 小 写 

/执行 SQL 语句 

// 判 断 如 果 SQL 语 句 的 类 型 为 SELECT 

// 执 行 该 语句 ， 获 取 查 询 结果 

// 判 断 语句 是 否 执 行 成 功 

// 如 果 查 询 结 果 为 0， 或 者 执行 失败 ， 则 返回 false 


// 否 则 返回 查询 结果 的 数组 


}elseif ($sqltype—"update" || $sqltype—"insert" || $sqltype—"delete"){ 
/判断 如 果 SQL 语 句 类 型 不 为 select、 则 执行 如 下 操作 
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这 srs) 
Teturn true: /| 执行 成 功 返 回 tme 
else 
Tetum false: /是 否 返 回 false 
} 
} 
// 分 页 类 
class SepPage{ 
Var S$rs; 
Var $pagesize: 
Var $nowpage; 
Var $nowpages: 
Var $array; 
Var $conn; 
Var $sqlstr; 
function ShowDate($sqlstr.$conn,$pagesize,Snowpage){ // 定 义 方法 
if(!isset(Snowpage) || Snowpage—"") /判断 变量 值 是 否 为 空 
S$this->nowpage=10: /定义 每 页 输出 的 记录 数 
else 
S$this->nowpage=$nowpage; 
S$this->pagesize=$pagesize; // 定 义 每 页 输出 的 记录 数 
Sthis->conn=$conn; // 连 接 数 据 库 返 回 的 标识 
Sthis->sqlstr=$sqlstr; // 执 行 的 查询 语句 
Sthis->rs=$this->conn->PageExecute($this->sqlstr, $this->pagesize,$this->nowpage); 
@S$this->array=$this->rs->GetRows(): // 获 取 记 录 数 
这 count($this->array) 一 0 || $this->rs—false) 
return false: 
else 
Teturn S$this->array: 
function ShowPage($contentname,$utits,$anothersearchstr, $class, Spage){ 
$allrs=$this->conn->Execute(S$this->sqlstr); /执行 查询 语句 
Srecord=count($allrs->GetRows()):; // 统 计 记 录 总 数 
S$pagecount=ceil($record/$this->pagesize): // 计 算 共 有 几 页 


$str.=" 共 有 ".$contentname."&nbsp:".Srecord."&nbsp:".$utits."&nbsp: 每 页 显示 &nbsp:".$this->pagesize."&nbsp; 
"Sutits."&nbsp: 第 &nbsp:".$this->rs->AbsolutePage0."&nbsp: 页 / 共 &nbsp:".Spagecount"&nbsp: 页 ": 

$str.="&nbsp:&nbsp:&nbsp:&nbsp:": 

if(!$this->rs->AtFirstPage()) 


$str.="<a href=".$_SERVER['PHP_SELF']."?page=".$page."&pages=1".$anothersearchstr." 


class=".$class."> 首 页 </a>": 


else 


$str.="<font color-#555555> 首 页 </font>": 


$str.="&nbsp:"; 
if(!$this->rs->AtFirstPage()) 


$str.="<a 


href=".$_SERVER['PHP SELF']."?page=".$page."&pages=".($this->rs->AbsolutePage()-1).$anothersearchstr." 
class=".$class."> 上 一 页 </a>": 


else 


$str.="<font color-#555555> 上 一 页 </font>": 


S47 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


$str="&nbsp:": 
if(!$this->rs->AtLastPage()) 
$str.="<a 
href=".$_SERVERI['PHP SELF']."?page=".$page."&pages=".($this->rs->AbsolutePage()+1).$anothersearchstr." 
class=".$class."> 下 一 页 </a>"; 
else 
$str.="<font color='#555555> 下 一 页 </font>"; 
Sstr.="&nbsp:"; 
if(!$this->rs->AtLastPage()) 
$str.="<a href=".$ SERVER[PHP SELF']."?page=".$page."&pages=".$pagecount.$anothersearchstr." 
class=".$class."> 尾 页 </a>": 
else 
S$str.="<font color=#555555> 尾 页 </font>": 
这 count(Sthis->array) 一 0 | $this->rs—false) 
Tetum "": 
else 
return $str; 


} 


?> 


12.5.2 ”Smarty 模板 配置 类 文件 


在 Smarty 模板 配置 类 文件 中 配置 Smarty 模板 文件 、 临 时 文件 、 配 置 文件 等 的 文件 路 径 。 
system.smarty.inc.php 文件 的 代码 如 下 : 


例 程 02 ”代码 位 置 ， 光盘 \TM\12\ShoppingSystem\system\system.smarty.inc.php 


<?php 
require("smarty/Smarty.class.php"): // 调 用 Smarty 类 文件 
class SmartyProject extends Smarty{ // 定 义 类 ， 继 承 Smarty 父 类 
function SmartyProjectO{ /定义 方法 ， 配 置 Smarty 模 板 
Sthis->template_dir ="./"; // 指 定 模板 文件 存储 在 根 目录 下 
Sthis->compile_dir ="./system/templates_c/":; // 指 定编 译文 件 的 存储 位 置 
Sthis->config dir ="./system/configs/"; // 指 定 配 置 文 件 的 存储 位 置 
Sthis->cache_dir ="./system/cache/": // 指 定 缓存 文件 的 存储 位 置 
上 
?> 


12.5.3 ”执行 类 的 实例 化 文件 


在 system.inc.php 文件 中 , 通过 require 语句 包含 system.smarty.inc.php 和 system.class.inc.php 文件 ， 
执行 类 的 实例 化 操作 ， 并 定义 返回 对 象 。 完 成 数据 库 连 接 类 的 实例 化 后 ， 调 用 其 中 的 GetConnId0 方 法 
连接 数据 库 。system.inc.php 文件 的 代码 如 下 : 
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例 程 03 ”代码 位 置 光盘 \TM\12\ShoppingSystem\system\system inc.php 

<?php 

require("system.smarty.inc.php"): 

require("system.class.inc.php"); 

$connobj=new ConnDB("mysql"."localhost"."root","root"."db_shop".false): /| 数据库 连接 类 实例 化 


$conn=$connobj->GetConnId(): // 连 接 数 据 库 并 返回 连接 标识 
$admindb=new AdminDB(): /| 数据 库 操作 类 实例 化 
S$seppage=new SepPage():; // 分 页 类 实例 化 

Ssmarty=new SmartyProject(): //Smarty 模 板 配置 类 的 实例 化 
?> 


12.6.1 前 台 首 页 概述 


对 前 台 首 页 的 设计 一 般 没 有 多 少 实 质 的 技术 ， 主 要 是 加 载 一 些 功能 模块 ， 如 登录 模块 、 导 航 栏 模 
块 、 公 告 栏 模块 等 ， 使 浏览 者 能 够 了 解 网 站 的 内 容 和 特点 。 设 计 网 站 首页 的 关键 之 处 是 要 合理 地 对 页 
面 进行 布局 ， 既 要 尽 可 能 地 将 重点 模块 显示 出 来 ， 同 时 又 不 能 因为 页 面 凌乱 无 序 ， 而 让 浏览 者 无 所 适 
从 ， 产 生 反感 。 本 系统 前 台 首 页 的 运行 结果 如 图 12.15 所 示 。 


FEW | m2 | Wm | ENE | 人 英才 人 | TE | We 


图 12.15 前 台 首 页 运行 结果 
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12.6.2 ”前 台 首页 技术 分 析 


在 前 台 首 页 中 应 用 Switch 语句 与 Smarty 模板 中 的 内 建 函 数 include 设计 一 个 框架 页 面 ， 实 现 不 同 
功能 模块 在 首页 中 的 展示 。 

Switch 语句 在 PHP 动态 文件 中 使 用 ， 根 据 超 级 链接 传递 的 值 ， 包 含 不 同 的 功能 模块 。 

Include 标签 在 Smarty 模板 页 中 使 用 ， 在 当前 模板 页 中 包含 其 他 模板 文件 。 其 语法 如 下 : 


{include file="file name " assign=" " var=" "} 


参数 说 明 : file 指定 包含 模板 文件 的 名 称 ，assign 指定 一 个 变量 保存 包含 模板 的 输出 ，var 传递 给 
待 包含 模板 的 本 地 参数 ， 只 在 待 包 含 模板 中 有 效 。 


12.6.3 前台 首页 实现 过 程 


国 ”前台 首页 使 用 的 数据 表 : tb commo、tb public、tb_user、tb_ links 
前 台 首 页 的 具体 实现 步骤 如 下 : 
(1) 创建 index.php 动态 页 。 在 index.php 动态 页 中 ， 应 用 include_once0 语 句 包含 相应 的 文件 ， 应 
用 Switch 语句 ， 以 超 链接 中 参数 page 传递 的 值 为 条 件 进行 判断 ,实现 在 不 同 页 面 之 间 跳 转 。index.php 
的 关键 代码 如 下 : 
例 程 04 ”代码 位 置 ， 光盘 \TM\12\ShoppingSystem \index.php 


<?php 
session _start(): /初始 化 session 变 量 
require_once("system/system.inc.php"): // 获 取 数 据 库 连 接 、 管 理 和 模板 类 返回 的 对 象 
include_once "public.php": // 包 含 公告 模块 
include_once "links.php": // 包 含 友情 链 接 模 块 
if($_SESSION["member"]—""){ // 判 断 SESSION 变 量 的 值 是 否 为 空 
S$smarty->assign("member","F"); // 如 果 为 空 ， 则 为 模板 变量 赋值 为 F 
}else{ 
Ssmarty->assign("member".$_SESSION['member’]): // 否 则 将 SESSION 变 量 的 值 赋 给 模板 变量 
3 
if($_GET["page"]—""){ // 判 断 变 量 page 的 值 是 否 为 空 
Ssmarty->assign("page","F"): // 如 果 为 空 ， 为 模板 变量 赋值 为 F 
3 
switch($_GET["page"]) { // 应 用 Switch 语 句 ， 根 据 条 件 进行 判断 ， 实 现 页 面 跳 转 
case "hyzx": 
include_once "member.php": // 包 含 指定 的 PHP 文 件 


这 $_GET[action] 一 "modify){ ”// 根 据 action 的 值 ， 指 定 不 同 的 模板 页 
S$smarty->assign("switchs".modifypwd.html): /将 指定 的 模板 页 赋 给 模板 变量 
yelse{ 
S$smarty->assign("switchs".'membershow.html"): 
} 
break: 
case 'allpub": 
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include_once 'allpub.php': 
Ssmarty->assign("switchs"."allpub.html"): // 将 指定 的 模板 页 赋 给 模板 变量 
break: 
/省 略 了 部 分 代码 
default: 
include_once mominate.php': 
S$smarty->assign("switches"."nominate.html"): 
include_once newhot.php'; 
$smarty->assign("switchs","newhot.html"); 


} 
S$smarty->display(Cindex.html): // 指 定 模板 页 
?> 


(2) 创建 index.html 模板 页 。 在 模板 文件 mdex.html 中 应 用 Smarty 的 include 标签 调用 不 同 的 模 
板 文件 ， 生 成 静态 页 面 。 其 关键 代码 如 下 : 


例 程 05 ”代码 位 置 : 光盘 \TMN12\ShoppingSystemindex.html 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<t> 
<td colspan="2"> {include file="top.html"}</td> 
</tr> 
<t> 
<td width="216" align="left" valign="top"> 
{if Smember—"F"} 
{include file="login.html"} 
{else} 
{include file="info.html"} 
{i 
{include file="public.html"} 
{include file="links.html"} 
</td> 
<td width="634" height="720" align="center" valign="top"> 
{include file="search.html"} 
{include file=$switchs} 
{if S$page—F} 
{include file=$switches} 
{/if}</td> 
</t> 
</table> 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td> {include fle=buttom html} </td> 
</t> 
</table> 


[DD 说 明 : 本 系统 的 功能 较 多 ， 结 构 比较 复杂 ， 对 于 初学 者 来 说 学 起 来 可 能 会 比较 困难 。 所 以 ， 本 书 
将 系统 中 的 各 个 功能 模块 所 涉及 的 文件 (如 PHP、TPL、CSS、JS 等 ) 尽 可 能 都 单独 实现 。 
读者 在 学 习 其 中 某 个 模块 时 ， 可 以 将 相关 的 文件 统一 放 到 同一 个 目录 下 单独 测试 。 
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12.7 登录 模块 设计 


12.7.1 登录 模块 概述 


用 户 登 录 模 块 是 会 员 功能 的 窗口 。 匿 名 用 户 虽 然 也 可 以 访问 本 网 站 ， 但 只 能 进行 浏览 、 查 询 等 简 
单 操作 ， 而 会 员 则 可 以 购买 商品 ， 并 且 能 享受 超 低 价格 。 登 录 模 块 包括 用 户 注册 、 用 户 登录 和 找 回 密 
码 3 部 分 ， 其 运行 结果 如 图 12.16 所 示 。 


图 12.16 登录 模块 运行 结果 


渡 


12.7.2 登录 模块 技术 分 析 


在 登录 模块 中 涉及 两 项 关键 技术 ， 第 一 项 是 通过 Ajax 实现 无 刷新 验证 用 户 名 是 否 被 占用 ， 第 二 项 
是 通过 GD2 函数 生成 登录 的 验证 码 。 下 面 是 这 两 项 技术 的 具体 实现 方法 。 
(1) Ajax 技术 无 刷新 验证 用 户 名 是 否 被 占用 ， 其 关键 代码 如 下 : 


例 程 06 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\check.js 
必 fomm 为 传 入 的 表单 名 称 ， 本 段 代码 为 register 表 单 */ 
function chkname(form){ 
谨 如 果 name 文 本 域 的 信息 为 空 名 为 name1 的 div 标 签 显示 如 下 信息 */ 
if(form.name.value—""){ 
namel.innerHTML="<font color=#FF0000> 请 输入 用 户 名 ! </font>": 
}else{ 
店 ”否则 获取 文本 域 的 值 */ 
var user = form.name.value: 
店 ” 生 成 rl 链接， 将 user 的 值 传 到 chkname.php 页 进行 判断 */ 
var url = "chkname.php?user="+user: 
信使 用 xmlhttprequest 技 术 运行 页 面 */ 
xmilhttp.open("GET".url,true): 
Xmlhttp.onreadystatechange = function|){ 
if(xmlhttp.readyState 一 4){ 
上 # ”根据 不 同 的 返回 值 ， 在 div 标 签 中 输出 不 同 信息 */ 
var msg = xmlhttp.responseText: 
iftmsg — 3){ 
namel.innerHTMIL="<font color=#FF0000> 用 户 名 被 占用 ! </font>":; 
Tetum false: 
jelseifmsg — 2){ 
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namel.innerHTMIL="<font color=green> 恭 喜 您 ， 可 以 注册 !</fonf>": 
片 ” 如 果 用 户 名 正确 ， 则 将 隐藏 域 的 值 改 为 "yes” */ 
form.c name.value = "yes": 
jelse{ 
namel.innerHTMIL="<font color=green> 未 知 错误 </font>"; 
} 
> 
lb 
xmlhttp.send(null): 


} 

在 该 函数 中 调用 chkname.php 页 ， 该 页 在 会 员 登 录 时 也 会 被 调用 ， 所 以 这 里 分 为 两 种 情况 ， 有 密 
码 和 无 密码 。 无 密码 为 注册 验证 ， 当 没有 返回 结果 时 ， 说 明 该 用 户 名 可 用 ; 而 有 密码 为 登录 验证 ， 和 
无 密码 相反 , 只 有 查询 记录 存在 时 , 才 允 许 登 录 , 并 将 用 户 名 和 用 户 ID 存储 到 session 中 。chkname.php 
页 面 的 代码 如 下 : 

例 程 07 ”代码 位 置 ; 光盘 \TM\12\ShoppingSystem\ chkname.php 


<?php 

session_start() /开启 session 支 持 
include_once 'system/system.inc.php'; // 加 载 数据 库 连 接 文件 
Sreback = '0' /声明 返回 变量 ， 初 值 为 “0 
$sql = "select * from tb_user where name='".$_GET[mser]."": // 生 成 sql 语 句 


雍 ”下 面 的 SQL 语 名 加 上 了 对 密码 的 比较 ， 这 是 用 来 验证 登录 的 ， 因 为 登录 验证 使 用 的 也 是 本 页 面 。*/ 
$password = $_GET[password']: 
这 !empty($password)){ 

$sql = "and password =".mds($password)."": /如 果 密 码 不 为 空 ， 说 明 是 登录 验证 


} 
启 ”执行 SQL 语 句 */ 
S$rst = $conn->Execute($sql) or die(execute error’): /根据 name 字 段 查询 数据 库 


这 Srst->RecordCountO — 1){ /判断 name 是 否 被 使 用 

Sreback ='3'; // 被 使 用 返回 3 
人 # 这 里 也 是 对 登录 页 面 进行 的 操作 ， 当 用 户 名 和 密码 输入 正确 时 ， 将 用 户 名 和 ID 存 进 session 中 #/ 
if(!empty($password)){ 


$_SESSION[member] = $rst->fields['name']: 
$_SESSION['id'] = Srst->fields['id]: 
Sreback = "1'; 
} 
}else{ 
Sreback = "2'; // 未 被 使 用 返回 2 
} 
echo $reback: 
?> 


(2) GD2 函数 库 生 成 验证 码 ， 其 关键 代码 如 下 : 
例 程 08 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\ yzm.php 


<2php 
srand((double)microtime()*1000000): 
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$im=imagecreate(42.16): 
S$black=imagecolorallocate($im,0.0.0): 
$white=imagecolorallocate($im.255.255.255): 
$gray=imagecolorallocate($in.200.200.200): 
imagefill($im.0.0.$gray): 
for($i=0:$i<4:$Si+H){ 
S$str=mt_rand(1,3); 
Ssize=mt rand(3.6); 
Sauthnum=substr($_GET[num],$i,1): 
imagestring($im. $size,(2+$i*10),$str,$authnum,imagecolorallocate($im.rand(0,130).rand(0,130).rand(0,130))): 
} 
for($i=0:$i<200:$iH+){ 
Srandcolor=imagecolorallocate($im,rand(0,255).rand(0,255).rand(0,255)):; 
imagesetpixel($im.rand(09670.rand09630.S$Srandcolon): 
} 
imagepng($im); 
imagedestroy($im); 
?> 


12.7.3 用 


户 注册 


国 用 户 注册 使 用 的 数据 表 : tb_user 


用 户 注册 
该 用 户 信 息 保 
所 示 。 


页 面 的 主要 功能 是 实现 新 用 户 注册 。 如 果 用 户 信息 输入 完整 而 且 符合 要 求 ， 则 系统 会 将 
存 到 数据 库 中 ， 和 否则 显示 错误 原因 ， 以 便 用 户 改正 。 用 户 注册 页 面 的 运行 结果 如 图 12.17 


身份 证 号 : [220204198seesas 输入 正确 
移动 电话 : [1351440ees* 联系 电话 只 能 由 数字 组 成 
固定 电话 : [043112345678 。 输 大 正确 
0 号码: [12345678 输入 正确 
ee 输入 正确 
辆 入 正确 
* 输入 正确 


输入 正确 


图 12.17 注册 模块 页 面 


用 户 注册 模块 的 具体 实现 步骤 如 下 : 


(1) 创 迁 


E register.html 模板 文件 ， 编 写 用 户 注 册页 面 。 其 中 包含 两 个 JS 脚本 文件 createxmlhttp js 
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和 check.js。 其 中 ，createxmlhttp.js 是 Ajax 的 实例 化 文件 ， 而 checkjjs 对 用 户 注册 信息 进行 验证 ， 并 且 
返回 验证 结果 。 
(2) 创建 register.php 动态 PHP 文件 ， 加 载 模板 。register.php 文件 的 代码 如 下 : 
例 程 09 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\register.php 


<2?php 
include_ once 'system/system.inc.php': // 载 入 配置 文件 
$smarty->assign('title',' 新 用 户 注册 '): // 设 置 title 
S$smarty->display('‘register.html"): // 显 示 register.html 
?> 


(3) 创建 reg_chk.php 文件 ， 获 取 表 单 中 提交 的 数据 ,将 数据 存储 到 指定 的 数据 表 中 。reg_chk.php 
文件 的 代码 如 下 : 


例 程 10 ”代码 位 置 光盘 \TM\12\ShoppingSystem\reg_chk.php 


<?php 
session_ start(); // 开 启 session 支 持 
include_once 'system/system.inc.php'; // 加 载 数据 库 连 接 文 件 


上 ”获取 表单 信息 */ 

$name =$_POST['name']: 

S$password = md5($_POST['pwd1"]); // 使 用 md5 给 密码 加 密 

S$question = $_POST['question"]: 

Sanswer = $_ POST['answer]: 

Srealname = $_POST['realname']: 

Scard = $_POST['card']; 

Stel =$_POST['tel]; 

Sphone =$_POST['phone']; 

SEmail = $_POST['email]: 

$QQ=$ POST[qq'1: 

S$code = $_POST['code']: 

Saddress =$_POST['address']: 

S$addtime = $conn->DBDate(time()): // 使 用 DBDate0 函 数 生成 当前 时 间 

/六 交趾 四 轴 中 中 和 市 办 灾 中 窒 中 兴 中 办 灾 中 字 中 昌 系 兴业/ 

久生 成 SQL 语句 */ 

$sql = " insert into tb_user( name, password, question, answer, realname, card, tel. phone, Email QQ, code, address, 
addtime ) ": 

$sql .= " values (‘$name', '$password', '$question'. '$answer', '$realname', '$card', '$tel', '$phone’, '$SEmail', '$QQ', 
'$code', '$address'.$addtime)": 

谨 执行 SQL 语句 */ 

Srst = $conn->execute($sql): 


if(Srst — false){ 
echo '<script>alert(\ 添 加 失败 \):history.back;</script>': 
}else{ 


$_SESSION['member'] = $name; 
S$_SESSION['id] = $conn->Insert IDO: 
echo "<script>top.opener.location reload():alert( 注 册 成 功 ):window.closeO:</script>"; 


SSS 
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(4) 创建 “用 户 注册 ” 超 链接 。 当 用 户 单 击 前 全 的 县 四 本 | 扩 乌 时 ， 系 统 会 调用 js 的 onclick 事件 ， 
弹出 注册 窗口 ， 其 代码 如 下 : 
例 程 11 代码 位 置 : 光盘 \TM\12\ShoppingSystem\login html 
<a href="#" id="login" onclick="reg0"> 用 户 注册 <Ja> 
这 里 使 用 到 的 js 文件 为 js/login.js， 调 用 的 函数 为 reg0。 该 函数 的 代码 如 下 : 


例 程 12 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\login.js 


function regO{ 
window.open("register.php", ”blank", “width=500,height=450",false): /弹出 新 窗口 
} 


12.7.4 ”用 户 登录 


国 ”用户 注册 使 用 的 数据 表 : tb_user 
用 户 登 录 模块 的 运行 结果 如 图 12.18 所 示 ， 需 要 在 此 输入 用 户 名 、 密 码 和 验证 码 。 


图 12.18 用 户 登 录 页 面 


用 户 登 录 模 块 的 具体 实现 步骤 如 下 : 
(1) 创建 模板 文件 login.html， 完 成 用 户 登录 表单 的 设计 。 在 该 页 面 中 当 单 击 Submit 按钮 时 ， 系 
统 将 调用 lg0 函 数 对 用 户 的 登录 提交 信息 进行 验证 ,lg0 函 数 包含 在 js/loginjjs 脚本 文件 内 , 其 代码 如 下 : 


例 程 13 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\login.js 
function lg(form){ 
谨 ”检验 用 户 名 */ 
if(form.name.value—""){ 
alert(' 请 输入 用 户 名 ): 
form.name.focus0: 
Tetum false: 


} 

族 ”检验 密码 */ 

这 form.password.value 一 "" || form.password.value.length < 6){ 
alert(' 请 输入 正确 密码 ): 
form.password.focus(): 
Teturn false: 


刻 ”检验 验 证 码 */ 


if(form.check.value — ""){ 
alert(' 请 输入 验证 码 ): 


>? 
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form.check.focus(): 
Teturn false: 


b 
启 ”比较 验证 码 */ 
if(form.check.value != form.check?2.value){ 
form.check selectO: 
code(form): 
Teturn false; 


店 ”根据 输入 的 用 户 名 和 密码 生成 url */ 
Var user = form.name.value; 
Var password = form.password.value:; 
var url = "chkname.php?user="+user+"é&password="+password: 
入 ”使 用 xmlhttp 对 象 来 返回 验证 结果 */ 
xmlhttp.open("GET",url,true): 
Xmlhttp.onreadystatechange = functionO{ 
if(xmlhttp.readyState 一 4){ 
var msg = xmlhttp.responseText; 
族 ” 如 果 返 回 值 不 等 于 2， 说 明 用 户 名 或 密码 错误 */ 


这 msg !{='2){ 
alert( 用 户 名 或 密码 错误 !: 
form.password.select(): 
form.check.value ="; 
code(form): 
}else{ 
谨 ” 如 果 成 功 ， 重 新 刷新 本 页 。 */ 
alert( 欢 迎 光临 9 
location reloadO; 
} 
} 
) 
xmlhttp.send(null); 
Teturn false; 


} 


用 户 名 和 密码 是 在 chkname.php 页 面 中 被 验证 。chkname.php 在 12.7.2 节 中 已 经 介绍 ， 这 里 不 再 

重复 。 
(2) 创建 用 户 信 息 模 板 文件 info.html。 用 户 登录 成 功 后 ， 在 原 登 录 框 位 置 将 显示 用 户 信息 ， 用 户 

可 以 通过 “会 员 中 心 ”对 自己 的 信息 做 修改 ， 也 可 以 单 击 “ 查 看 购物 车 ” 超 链 接 查 看 购物 车 商品 ， 当 
用 户 离 开 时 可 以 单 击 “ 安 全 离开 ” 超 链 接 。 用 户 信息 模块 的 主要 代码 如 下 : 

例 程 14 代码 位 置 ， 光盘 \TM\12\ShoppingSystem\info.html 

<!-- 显示 当前 登录 用 户 名 --> 

欢迎 您 ，{Smember} 

<!-- 会 员 中 心 超 链接 -> 

<a href="?page=hyzx" id="info" class="lk"> 会 员 中 心 </a> 

<!-- ”查看 购物 车 -> 

<a hre 全 "2?page=shopcar" class="lk"> 查 看 购物 车 </a> 


Ecyh 
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<!-- 安全 离开 -> 
<a onclick="javascript:logout(" style="cursor:hand" id="info"> 安 全 离开 </a> 


12.7.5” 找 回 密码 


国 用户 注册 使 用 的 数据 表 : tb_user 

登录 模块 的 最 后 一 个 部 分 就 是 找 回 密码 。 找 回 密码 是 根据 用 户 在 填写 资料 时 所 填写 的 密 保 问 题 和 
密 保 答案 来 实现 的 。 当 用 户 单 击 “ 找 回 密码 ” 超 链接 时 ， 首 先 提示 输入 要 找 回 密码 的 会 员 名 称 ， 然 后 
根据 密 保 问题 填写 密 保 答案 ， 最 后 重新 输入 密码 。 找 回 密码 模块 的 流程 如 图 12.19 所 示 。 


http: //1localhos el http: /localhos 可 


找 加 密码 
密 保 问题 :hidepengyou 


me 


[而 
[二 


元 成 


第 三 步 : 更 新 密码 第 四 步 : 更 改 成 功 
图 12.19 找 回 密码 流程 图 


1， 创 建 模板 文件 


虽然 找 回 密码 需要 4 个 步骤 ， 但 实际 上 每 个 步骤 使 用 的 都 是 相同 的 模板 文件 和 js 文件 ， 只 是 被 调 
用 的 表单 和 js 函数 略 有 差别 。 这 里 根据 不 同 的 文件 来 分 别 进行 介绍 。 
该 模板 文件 一 共 包含 了 3 个 表单 ， 分 别 代表 了 3 个 步骤 ， 其 核心 代码 如 下 : 


例 程 15 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\found.html 
<!--” 载 入 两 介 js 脚 本 文件 --> 
<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/found.js"></script> 
<!-- 第 1 个 div 标 签 --> 
<div id="first"> 
<table width="200" border="0" cellspacing="0" cellpadding="0"> 
<form id="foundname" name= "found" method="post" action="#"> 

<tr><td> 找 回 密码 </td></t> 

<tr><td> 会 员 名 称 : </td> 

<!-- ”text 文 本 域 ， 用 于 输入 要 找 回 密码 的 会 员 名 称 ”--> 

<td><input id="user" name="user" type="text" class="txt"></td> 
</t> 
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<tr><td> 
<!-- 单 击 “ 下 一 步 ”按钮 ， 能 触发 onclick 事 件 来 调用 chkname 函 数 ”--> 
<input id = "nextl " name = " nextl " type = " button " class = " bm " value = ”下 一 步 " onClick = " retum chkname 
(foundname ) "/></td></t> 
</form> 
</table> 
</div> 
<!-- 第 2 个 div 标 签 ， 样 式 为 隐藏 --> 
<div id="second" style="display:none:"> 
<table> 
<form id="foundanswer" name="found" method="post" action="#"> 
<tr><td > 找 回 密码 </td></tr> 
<t><td> 密 保 问题 : </td> 
<!-- ”用 于 显示 密 保 问题 的 div 标 签 -~> 
<td <div id="question"></div></td></tr> 
<tr><td> 密 保 答案: </td> 
<!-- 文本 域 ， 用 于 填写 密 保 答案 -~ 
<td ><input id="answer" name="answer" type="text" class="txt" /></td></tr> 
<t> 
<!-- 单 击 “ 下 一 步 ” 按 钮 ， 用 来 触发 onclick 事 件 ， 并 调用 chkanswer0 函 数 ”--> 
<td><input id = " next2 " name = " next2 " type=" button " class=" bm " value =" 下 一 步 " onClick = " retum chkanswer 
(foundanswer ) "></td> 
</u> 
</form> 
</table> 
</div> 
<!-- ”第 3 个 div 标 签 ， 样 式 也 为 隐藏 ， 作 用 是 修改 密码 -> 
<div id='third' style="display:none:"> 
<table> 
<form id="modifypwd" name="found" method="post" action="#"> 
<tr><td> 输入 密码 </td></t> 
<tr><td> 输 入 密码 :</td> 
<td><input id="pwd1" name="pwd1" type="password" class="txt"></td></tr> 
<tr><td> 确 认 密码 :</td> 
<td><input id="pwd2" name="pwd2" type="password" class="txt" /></td> 
</t> 
<t> 
<!-- 单 击 “ 完 成 ”按钮 ， 调 用 ckpwd0 函 数 --> 
<td><input id =" mod " name =* mod " type = " button " class =" btn " value =" 完成 " onClick = "retum chkpwd 
(modifypwd) "></td> 
</u> 
</form> 
</table> 
</div> 


可 以 看 出 ， 在 上 述 3 个 表单 中 ， 只 有 一 个 表单 默认 情况 下 是 显示 的 ， 其 他 均 为 隐藏 。 只 有 通过 调 
用 不 同 的 js 函数 ， 才 可 以 对 其 他 表单 进行 操作 。 
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2. 创建 js 脚本 文件 

foundjs 脚本 文件 包含 3 个 函数 : chkname0、chkanswer0 和 chkpwd0。 其 中 ，chknameO 函 数 的 作 
用 是 检查 用 户 输入 的 会 员 名称 是 否 存在 ， 如 果 存 在 ， 则 使 用 xmlhttp 对 象 去 调用 生成 的 url 进行 处 理 判 
断 。 如 果 该 用 户 存在 ， 则 隐藏 当前 表单 ， 并 显示 下 一 个 表单 ， 最 后 输出 密 保 问题 。chknameO 函 数 的 代 
码 如 下 : 


例 程 16 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\found.js 
function chkname(form){ 

族 ”获取 文本 框 信息 */ 

Var user = form.user.value; 

语 ”如 果 为 空 ， 则 输出 提示 。 */ 

ifuser 一 ")f 
alert(' 请 输入 用 户 名 '); 
form.user.focus():; 

Teturn false: 

}else{ 

谨 ”否则 ， 生 成 url， 并 调用 xmlhttp 对 象 ” */ 
var ur] = "foundpwd.php?user="+user; 
xmlhttp.open("GET",url,true): 
xmlhttp.onreadystatechange = function(O){ 
if(xmlhttp.readyState 一 4){ 

var msg = xmlhttp.responseText; 

让 ”如 果 没 有 结果 ， 则 提示 。*/ 

if(msg —'0){ 
alert(' 没 有 该 用 户 ， 请 重新 查找 '"); 
form.user.selectO: 

Tetum false; 

jelse{ 

谨 ” 否则， 隐藏 当 前 表单 ， 显 示 下 个 表单 ， 并 输出 密 保 问题 */ 
document.getElementById('first).style.display = ‘hone'; 
document.getElementById('second').style.display = 
document.getElementById('question'").innerHTML = msg: 

} 

上 
b 
xmilhttp.send(null): 


} 


其 他 两 个 函数 也 使 用 xmlhttprequest 对 象 ， 实 现 方法 相差 无 几 ， 不 同 之 处 就 是 对 返回 值 的 处 理 ， 
chkanswerO 函 数 隐 藏 当前 表单 ， 显 示 下 一 个 表单 。chkanswerO 函 数 的 代码 如 下 : 


例 程 17 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\found.js 
function chkanswer(form) { 
必 ”获取 已 隐藏 表单 数据 */ 

var user = document.getElementById(Cuser).value: 

请” 获取 密 保 答案 */ 


Var answer = form.answWer.Value: 
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全 ”表单 验证 *#/ 


这 answer — "){ 


}else{ 
从 ”生成 urlf */ 
var url = "foundpwd.php?user="+user+"&answer="+answer:; 
Xmlhttp.open("GET".urLtme): 
Xmlhttp.onreadystatechange = fonction0{ 
这 xmlhttp readyState 一 4){ 
var msg = xmlhttp.responseText: 
这 msg 一 '0){ 
alert( 问 题 回答 错 误 ): 
form.answer.select(): 
Tetum false; 
}else{ 
document.getElementById(second').style.display = "none'; 
document.getElementById(Cthird).style.display = "; 


}; 
} 
Xmlhttp.sendCnulD: 


} 


而 chkpwd0 函 数 则 提示 用 户 操作 状态 ， 如 果 成 功 ， 则 关闭 当前 页 。ckpwd0 函 数 的 代码 如 下 : 


例 程 18 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\found.js 
function chkpwd(form) { 
族 ”获取 名 称 和 密码 */ 
Var user = document.getElementById(user).value: 
var pwdl = formpwdl.value: 
Var pwd2 = form.pwd2.value: 
店 ”省 略 部 分 为 表单 验证 */ 
var url = "foundpwd.php?user="+user+"&password="+pwd1: 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = functionO{ 
if(xmlhttp.readyState 一 4){ 
varmsg=Xmlhttp.responseText: 


这 msg —'1){ 
alert( 密 码 修改 成 功 ， 请 重新 登录 ): 
window.close(); 
b 
b 
xmlhttp.send(null): 


} 
3. 创建 数据 处 理 文件 


foundpwd.php 文件 的 功能 是 根据 用 户 输入 信息 来 检测 数据 表 中 的 数据 ， 并 根据 不 同 的 输入 信息 返 
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回 


不 同 的 结果 。 该 文件 的 代码 如 下 : 
例 程 19 代码 位 置 光盘 \TM\12\ShoppingSystem\foundpwd.php 


<?php 
include_once 'system/system.inc.php': // 载 入 数据 库 连 接 文 件 
S$name= $_GET[user]: // 获 取 表 单 值 


Sanswer = $_GET['answer']: 
Spassword = $ GET[password']: 
Sreback = "0'; 
店 ”如 果 变 量 Sanswer 和 S$password 为 空 ， 说 明 用 户 只 输入 了 会 员 名 称 “*/ 
这 empty(Sanswern) &é&: empty(Spassword)) { 
久生 成 查询 语句 */ 
Snamesql = "select * from tb_user Where name 王 ".S$name.""; 
作 。 获 取 查 询 结果 */ 
Snamerst = $conn->execute($namesq]); 
必 “ 如 果 查 询 记录 为 1， 说 明 输入 正确 */ 
这 $namerst->recordCountO — 1){ 
上 汪 ” 获 取 密 保 问题 ， 并 赋 给 变量 Sreack */ 
$question = $namerst->fields['question ]: 
Sreback = $question: 


} 
让 ”如 果 变 量 Sanswer 不 为 空 ， 说 明 用 户 输入 了 密 保 答案 yy 
}else if(!empty($answer)){ 
族 ”根据 密 保 答案 和 会 员 名 称 ， 生 成 查询 语句 。 */ 
Sanswersql] = "select * from tb_user Where name = ".$name."" and answer = ".$answer.""; 
谨 ”返回 查询 结果 ， 并 判断 是 否 有 返回 记录 ， 如 果 有 ， 说 明 输入 正确 */ 
Sanswerrst = $conn->execute($answersql): 
if($answerrst->recordCount| 一 1){ 
Sreback = "1'; 


} 
让” 如 果 变 量 $Spassword 不 为 空 ， 说 明 用 户 输入 了 新 密码 */ 
}else if(!empty($password)){ 
雍 ”根据 用 户 名 称 和 新 密码 更 新 记录 。*/ 
$sql = "select * from tb_user where name = ".$name."™"; 
$arr = array(); 
Srst = $conn->execute($sq]); 
if($rst->RecordCountO| =— 1){ 
$arr['password'] = md5($password): 
SupdateSQL = $conn->GetUpdateSQL(S$rst,$arr.,true): 
$conn->execute($updateSQL); 
Sreback = "1'; 
} 


睛 ”输出 返回 值 */ 
echo S$reback: 
?> 


4. 加 载 模板 页 
因为 所 有 登录 模块 的 模板 都 不 需要 或 者 只 需要 传递 一 两 个 变量 ,所 以 PHP 加 载 页 的 内 容 比 较 简 单 。 
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找 回 密码 页 面 的 代码 如 下 : 
例 程 20 ”代码 位 置 光盘 \TM\12\ShoppingSystem\found.php 
<?php 
include once 'system/system.inc.php': // 载 入 配置 文件 
Ssmarty->assign('"title', 找 回 密码 '): /向 模板 中 传递 title 变 量 
S$smarty->display(found html"): /显示 found html 模 板 页 
?> 


信息 模块 设计 


忆 


12.8 


12.8.1 会 员 信息 模块 概述 


ba ddim dn 查看 
“安全 退出 ”进行 讲解 ， 关 于 


te 12.20 所 示 。 


用 户 登 录 后 ， 即 可 看 到 会 员 信息 模块 。 在 这 里 ， 
购物 车 和 安全 退出 等 操作 。 本 节 只 对 会 员 信息 模块 中 的 “会 员 中 心 ” 
“查看 购物 车 ”模块 将 在 商品 模块 中 进行 介绍 。 会 
条 会 员 中 心 inoider | 


上 2) 查看 傅 息 ?>> 侍 改 室 民 


会 员 篇 号 : | 23 
会 员 名 称 : | or 


和 从 证 号 :| 7260555551105665 | 
移动 电话 | |13604338764 + 
Be | 
Email :| Parkaihual 103sina. col 
WW: 546827366 
六 :| 130025 
Mts 
CE aa| 


图 12.20 会 员 中 心 


12.8.2 会 员 信息 模块 技术 分 析 


在 会 员 信息 模块 中 , 以 SESSION 变量 中 存储 的 用 户 名 称 为 条 件 , 从 会 员 信息 表 中 查询 出 会 员 信息 ， 
并 且 将 信息 存储 到 模板 变量 中 ， 最 后 在 模板 页 中 输出 。Member.php 的 代码 如 下 : 
例 程 21 代码 位 置 : 光盘 \TM\12\ShoppingSystemmember.php 


<2php ee 
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Session start(): /初始 化 SESSION 变量 
include_ once 'system/system inc php': /包含 配置 文件 
人 # ”查找 用 户 资料 所 
$sql = "select * from tb_user Where name =".$ SESSION['member]."":; /执行 查询 操作 
Srst = $conn->execute($sq]): 
Sarr = $rst->GetArrayO; 
Ssmarty->assign('arr’, Sarr[0]): // 将 查询 结果 赋 给 模板 变量 
?> 


Menber.php 文件 中 查询 出 的 数据 是 会 员 信息 模板 功能 实现 的 根本 。 
12.8.3 会 员 中 心 


国 ”用户 注册 使 用 的 数据 表 : tb_user 

当 单 击 “ 会 员 中 心 ” 超 链 接 时 ， 会 回 传 给 当前 页 一 个 page 值 ， 当 前 页 根据 这 个 page 值 来 载 入 
member.php 文件 。 

1. 创建 PHP 页 面 

与 登录 模块 的 设计 不 同 ， 本 节 首 先 来 创建 PHP 页 面 。 因 为 该 模块 中 的 模板 需要 使 用 数据 库 中 的 数 
据 及 一 些 动态 信息 ， 这 些 都 需要 在 PHP 页 中 先行 获取 及 处 理 ， 然 后 再 传 给 模板 页 。 会 员 中 心 页 面 的 代 
码 请 参考 技术 分 析 中 的 内 容 。 

2. 创建 模板 页 

该 模块 包括 查看 信息 模板 及 修改 密码 模板 。 首 先 介绍 查看 信息 模板 ， 代 码 如 下 : 


例 程 22 ”代码 位 置 ; 光盘 \TM\12\ShoppingSystem\membershow.html 
<!-- 显示 超 链接 ”--> 
<p>{$smarty.session.member}<a href =' ?page=hyzx ' > 查看 信息 </a><a href = ' ?page = hyzx&action=modify' 
id="mem"> 修 改 密码 </a></p> 
<table> 
<form id = "member " name = " member " method = " post " action = ”modify info_chkphp " onsubmit = " retum 
mem( member ) " > 
<!-- ”显示 当前 会 员 信息 ， 此 处 为 不 可 更 改 信息 --> 
<tr><td colspan="2">{$arr.name} 信 息 ( 不 可 更 改 信息 ) </td></t> 
<!-- ”因为 格式 相同 ， 所 以 只 显示 一 条 ， 其 他 省 略 --> 
<tr><td> 会 员 编号 : </td><td >&nbsp; {$arr.id}</td></t> 


<tr><td>{$arrname} 信 息 〈 可 更 改 信 息 ) </td></tr> 
<!-- 可 以 修改 的 信息 ， 放 到 表单 元 素 中 --> 
<tr><td> 真 实 姓名 : </td> 
<td><input id="realname" name="realname" type="text" value=" {S$arr.realname}" /></td></tr> 
<tr><td> 身 份 证 号 :</td> 
<td><input id="card" name="card" type="text" value=" {$arr.card}" /></td></tr> 
<tr><td> 移 动 电话 : </td> 
<td><input id="tel" name="tel" type="text" value=" {$arr.tel}"></td></tr> 
<tr><td> 固 定 电话 : </td> 
<td><input id="phone" name="phone" type="text" value=" {$arr.phone}" /></td></t> 
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<tr><td>Email: </td> 
<td><input id="email" name="email" type="text" value=" {$arr.Email}" /></td></tr> 
<tr><td>QQ 号 : </td> 
<td><input id="qq" name="qq" type="text" value="{$arr.QQ}" /></td></t> 
<t> 
<td> 邮 编 : </td> 
<td><input id="code" name="code" type="text" value=" {$arr.code}" /></td></tr> 
<tr><td> 地 址 :</td> 
<td><input id="address" name="address" type="text" value=" {$arr.address}" /></td></t> 
<t> 
<td><input name="enter" type="submit" id="enter" value=" 修 改 " /> 
<input name="reset" type="reset" id="reset" value=" 重 置 " /></td></tr> 
</form> 
</table> 


修改 密码 模板 比 修改 信息 模板 要 简单 一 些 ， 因 为 不 需要 额外 地 传 值 。 模 板 代码 如 下 : 


例 程 23 ”代码 位 置 ， 光盘 \TM\12\ShoppingSystem\modifypwd.html 
<!-- 载 入 脚本 文件 --> 
‘<script language="javascript" src="js/member.js"></script> 
<p> {$smarty.session.member}<a href-' ?page=hyzx> 查 看 信息 </a><a hre 伍 ?page=hyzx&action=modify' > 修改 密码 
</a></p> 
<table> 
<!-- ”修改 密码 表单 ”--> 
<form id = "member " name = " member " method = " post " action = " modify pwd_chk.php " onsubmit = " return pwd 
(member) "> 
<tr><td > 修改 密码 </td></h> 
<tr><td > 原 密码 : </td><td><input id="old" name="old" type="password" /></td></tr> 
<tr><td> 新 密码 : </td><td><input id="new1" name="newl" type="password" /></td></tr> 
<tr><td> 确 认 密 码 : </td><td><input id="new2" name="new2" type="password" /></td></tr> 
<tr><td><input id="enter' name="enter" type="submit" value=" 修 改 " /></td></tr> 
</form> 
</table> 


3. 创建 脚本 文件 

该 模块 的 脚本 文件 和 用 户 注 册 模块 类 似 ， 都 是 对 信息 的 合法 性 进行 验证 ， 如 信息 是 否 为 室 、 是 否 
符合 规范 等 ， 这 里 不 再 歼 述 。 

4. 创建 处 理 页 


当 信 息 验证 通过 后 ， 系 统 将 跳 转 到 处 理 页 进行 信息 处 理 。 本 模块 处 理 页 分 为 信息 修改 和 密码 修改 
两 个 页 面 。 首 先 介绍 信息 修改 页 ， 代 码 如 下 : 


例 程 24 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\modify_info_chk.php 


<?php 
session start():; // 开 启 session 支 持 
include_once ‘system/system.inc.php’: // 载 入 数据 库 连 接 文 件 
S$sql ='select * from tb_user where id =".$_SESSIONT'id']: // 根 据 id 生 成 sq 查询 语句 
Srst = $conn->execute($sq]): // 执 行 sql 语 句 ， 返 回 查 询 结果 集 
Smod = aray(: // 声 明 数组 


启 ”将 修改 信息 添加 到 数组 中 */ 
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S$mod[realname]=$_ POST[realname']: 
S$mod['card]=$ POST['card]: 
S$mod[tel]=$_POST['tel]: 
S$mod[phone] = $_POST['phone']: 
S$mod[Email] = $_ POST['email]: 
Smod[rQQ]=S$ POST[qql]: 
Smod['code'] = $_POST['code']: 
Smod['address'] = $_POST['address']; 
/二村 训 训 于 环 束 于 可 这 于 于 这 环球 训 字 于 训 束 事 机 束 可 可 束 束 机 束 字 字 守 可 字 事 字 字 
SupdateSQL = Sconn->GetUpdateSQL(SrsLSmod); // 生 成 更 新 语句 
if($conn->execute($updateSQL)) 
echo "<script>alert( 修 改 成 功 '):location=(index.php);</script>"; // 收 改 成 功 ， 跳 到 首页 
else 
echo "<script>alert( 修 改 失败 "):history.go(-1):</script>": /否则 ， 返 回 上 一 页 
?> 


密码 修改 页 的 操作 流程 也 十 分 类 似 ， 只 是 更 新 的 数组 要 小 得 多 ， 只 有 一 个 字段 。 修 改 密码 页 的 代 
码 如 下 : 


例 程 25 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\modify_pwd_chk.php 


<?php 
session_start(); // 开 启 session 支 持 
include_once 'system/system.inc.php':: // 载 入 数据 库 连 接 文 件 
$oldpwd=mds($_ POST['old]): /给 密码 进行 mds 加 密 
人 # 根据 it 和 mds 加 密 后 的 密码 查询 用 户 bh 
$sql = 'select * from tb_user Where id =".$_SESSION["id'].' and password =\".$oldpwd.\"; 
S$rst = $conn->execute($sq]): // 返 回 查 询 结果 
S$mod = array0; // 创 建 更 新 数组 
Smod['password'] = md5($_POST['new1")): // 为 新 密码 进行 md5 加 密 
SupdateSQL = $conn->GetUpdateSQL($rst, Smod); // 生 成 更 新 语句 


if($conn->execute($updateSQL)) 
echo "<script>alert( 修 改 成 功 "):location=('index.php"):</script>"; // 修 改 成 功 ， 返 回首 页 
else 
echo "<script>alert( 修 改 失败 "):history.go(-1):</script>": // 否 则 ， 返 回 上 一 页 
es 


12.8.4 安全 退出 


当 用 户 需要 离开 网 站 时 ， 可 以 单 击 “ 安 全 退出 ” 超 链 接 来 调用 logout0 函 数 ， 当 用 户 确认 退出 后 ， 
则 跳 转 到 logout 页 面 ， 销 毁 session 并 回 到 首页 。 安 全 退出 所 涉及 的 页 面 代码 如 下 : 
例 程 26 ”代码 位 置 ， 光盘 \TM\12\ShoppingSystem\js\info.js 


function logoutO{ 
这 confimm(" 确 定 要 退出 登录 吗 ? ")){ /输出 选择 框 ， 用 户 可 以 单 击 “确认 ”或 “取消 ”按钮 
window.open(logoutphp'… parent.".false): /如 果 用 户 确认 退出 ， 则 打开 logoutphp 页 
}else 
Teturn false: 


第 12 章 电子 商务 网 站 ( Apache+PHP+ADODB+Smarty+Ajax+MySQL 实现 ) 


例 程 27 代码 位 置 光盘 \TM\12\ShoppingSystem\logout.php 


<?php 
Session start(): /开启 session 支 持 
Session destroy0: /销毁 session 
echo '<script>alert(' 用 户 已 安全 退出 I\):location=(\'index.php\):</script>'; // 回 到 首页 

?> 


12.9 商品 显示 模块 


12.9.1 商品 显示 模块 概述 


本 系统 为 用 户 提供 了 不 同 的 商品 展示 方式 ， 包 括 推荐 商品 、 最 新 商品 、 热 门 商品 等 ， 能 够 使 消费 
者 有 目的 地 选 购 商品 。 每 个 展示 方式 中 均 包 括 商品 的 详细 信息 显示 ， 为 用 户 购买 商品 提供 了 可 靠 的 依 
据 。 本 系统 商品 显示 模块 的 运行 结果 如 图 12.21 所 示 。 


数码 相机 
市 场 价 : 1900 元 
会 员 价 ; 1620 元 


图 12.21 商品 展示 模块 页 面 


因为 推荐 商品 、 最 新 商品 和 热门 商品 的 实现 方法 和 过 程 基本 相同 ， 所 以 本 节 只 讲解 推荐 商品 模块 。 
其 他 功能 相关 代码 读者 可 参见 光盘 中 的 源 程序 。 


12.9.2 ”商品 显示 模块 技术 分 析 


商品 显示 功能 实现 的 关键 就 是 如 何 从 数据 库 中 读 取 商 品 信息 和 如 何 完成 数据 的 分 页 显示 。 在 定义 
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SQL 语句 时 ， 首 先 判断 字段 isnom 的 值 ， 如 果 值 为 1， 即 为 推荐 ， 否 则 为 不 推荐 ， 然 后 再 定义 数据 降 
守 排 列 ， 并 设置 每 页 显示 4 条 记录 ， 这 就 是 完成 商品 显示 的 查询 语句 ， 其 代码 如 下 : 
例 程 28 ”代码 位 置 : 光盘 \TM\12\ShoppingSystemnominate.php 


<?php 
include_ once 'system/system_inc .php'- // 连 接 数据 库 
$sql = "select idname.pics.m price.v price from tb commo where isnom = 1 order by id desc limit 4": 
Snomarr = $admindb->ExecSQL($sql.$conn):; // 执 行 查询 语句 
Ssmarty->assign('nomarr’, $nomarr): // 将 查询 到 的 数据 赋 给 模板 变量 
?> 


最 后 ， 定 义 模板 文件 ， 并 通过 section 语句 循环 输出 存储 在 模板 变量 中 的 数据 ， 即 完成 商品 展示 的 
操作 。 
section 是 Smarty 模板 中 的 一 个 循环 语句 ， 该 语句 用 于 复杂 数组 的 输出 。 其 语法 如 下 : 


{section name= "sec_name" loop=$arr_ name start=num step=num} 


参数 含义 : name 是 该 循环 的 名 称 ，loop 为 循环 的 数组 ，start 表示 循环 的 初始 位 置 ， 如 start=2， 那 
么 说 明 循 环 是 从 loop 数组 的 第 2 个 元 素 开始 的 ，step 表示 步 长 ， 如 step=2， 表 示 循 环 一 次 后 ， 数 组 的 
指针 将 向 下 移动 2 位 ， 依 此 类 推 。 


12.9.3 ”商品 显示 模块 的 实现 过 程 


国 ”商品 显示 使 用 的 数据 表 : tb_commo、tb_user 
在 技术 分 析 中 已 经 对 商品 显示 所 使 用 的 技术 、 方 法 进行 了 概述 ， 下 面 就 介绍 一 下 具体 的 实现 过 程 。 
(1) 创建 nominate.php 文件 ， 从 数据 库 中 读 取出 推荐 商品 的 数据 ， 并 将 数据 存储 到 模板 变量 中 ， 
其 代码 可 以 参考 技术 分 析 。 
(2) 创建 nominate.html 模板 页 ， 应 用 section 语句 输出 商品 信息 ， 并 添加 相应 的 操作 按钮 或 链接 。 
模板 页 中 一 共有 3 个 事件 ， 显示 更 多 商品 、 查 看 商品 和 放 入 购物 车 。 

当 单 击 “更 多 商品 ” 超 链接 时 ， 将 会 重新 加 载 本 页 面 ， 并 传递 一 个 page 变量 。switch 语句 会 
根据 page 值 来 显示 。 

当 单 击 “ 查 看 详情 ”按钮 时 ， 将 触发 onclick 事件 ， 并 将 调用 openshowcommo0 函 数 ， 同 时 ， 
商品 id 会 作为 函数 的 唯一 参数 被 传递 进去 。 

回 ” 当 单 击 “ 购 买 ”按钮 时 ， 同 样 会 触发 onclick 事件 ， 并 调用 buycommo0 函 数 ， 唯 一 的 参数 也 
是 商品 的 id。 

商品 模板 页 面 的 代码 如 下 : 

例 程 29 代码 位 置 : 光盘 \TM\12\ShoppingSystemnominate.html 

<!-- 载 和 css 样式 --> 

<link href="css/nominate.css" rel="stylesheet" type="text/css" /> 

<link href="css/links.css" rel="stylesheet" type="text/css" /> 

<!-- 载 入 js 脚本 文件 -二 

<script language="javascript" src="js/createxmlhttp.js"></script> 

‘<script language="javascript" src="js/showcommo.js"></script> 
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<table width="643" border="0" cellpadding="0" cellspacine="0"> 
<tr> 
<td colspan="6" backeround="images/default 14.gif' width="636" height="39" align="right" 
valien="middle"><a href="?page=nom" class="lk">&et:&et:more&lt:&lt:.</a></td> 
</t> 
<t> 
{section name=id loop=$nomarr} 
<td width="145" height="185" align="left" valign="top"> 
<table width="145" border="0" cellpadding="0" cellspacine="0" > 
<tr> 
<td height="100” align="center" valign="middle"><img src="{S$nomar[id].pics}” width="100" height="80" 
alt="{$nomarr[id].name}" style="border: 1px solid #f0f0f0;" ></td> 
</t> 
<t> 
<td height="32" align="center” valign="middle"><input id="showinfo" name="showinfo" type="button" 
value="" class="showinfo" onclick="openshowcommo( {$nomarr[id].id})"/>&nbsp;<input id="buy" name="buy" type="button" 
value="" class="buy" onclick="return buycommo( {$nomarr[id].id})" /></td> 


{/section} 
</t> 
</table> 


(3) 创建 showcommo.js 脚本 文件 。 当 单 击 “ 查 看 商品 ”按钮 时 ， 系 统 会 弹出 一 个 新 的 页 面 ， 并 
显示 商品 的 详细 信息 ; 当 单 击 “ 购 买 ”按钮 时 ， 该 商品 将 会 被 放 到 当前 用 户 的 购物 车 中 ， 如 果 用 户 没 
有 登录 或 商品 已 添加 ， 则 会 提示 错误 信息 。js 脚本 文件 的 代码 如 下 : 

例 程 30 ”代码 位 置 ; 光盘 \TM\12\ShoppingSystem\js\showcommo.js 

必 ”查看 商品 信息 函数 ， 将 打开 一 个 新 页 面 并 


function openshowcommo(key){ 
open('showcommo.php?id="+key,'_blank',"width=560 height=300",false): 


} 
必 ”将 购买 商品 添加 到 购物 车 中 ， 将 在 下 节 中 讲解 ”*/ 
function buycommo(key){ 


} 
12.10 ”购物 车 模块 设计 


12.10.1 购物 车 模块 概述 


购物 车 在 电子 商务 平台 中 是 前 台 客 户 端 程序 中 非常 关键 的 一 个 功能 模块 。 购 物 车 的 主要 功能 是 保 
留用 户 选 择 的 商品 信息 ， 用 户 可 以 在 购物 车 内 设置 选 购 商 品 的 数量 ， 显 示 选 购 商品 的 总 金额 ， 还 可 以 
清除 选择 的 全 部 商品 信息 ， 重 新 选择 商品 。 购 物 车 页 面 的 运行 结果 如 图 12.22 所 示 。 
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商品 名 称 购买 数量 


| 宗主 影院 I 
口 数码 相机 i 
许 选 反选 。 莉 除 渤 接 


12.22 ”购物 车 页 面 
购物 车 模块 主要 用 于 实现 添加 商品 、 删 除 商品 和 更 改 数量 等 操作 。 


12.10.2 ”购物 车 模块 技术 分 析 


实现 购物 车 功能 最 关键 的 部 分 就 是 如 何 添 加 商品 ， 如 果 不 能 完成 商品 的 添加 ， 那 么 购物 车 中 的 其 
他 操作 都 没有 任何 意义 。 
购物 车 模块 的 具体 实现 步骤 如 下 : 

(1) 在 商品 显示 模块 中 ， 单 击 商品 中 的 “购买 ” 按钮 ， 将 商品 放 到 购物 车 中 ， 并 进入 到 “购物 车 ” 
页 面 。 单 击 “ 购 买 ”按钮 调用 buycommo0 函 数 ， 购 买 商品 的 id 是 该 函数 的 唯一 参数 ， 在 buycommo0 
函数 中 通过 xmlhttp 对 象 调用 chklogin.php 文件 ， 并 根据 回 传 值 做 出 相应 处 理 。buycommo0) 函 数 的 代码 
如 下 : 

例 程 31 代码 位 置 ; 光盘 \TM\12\ShoppingSystem\js\showcommo.js 


* 添 加 商品 ， 同 时 检查 用 户 是 否 登录 、 商 品 是 否 重复 等 
dh 
function buycommo(key){ 
雍 ”根据 商品 DD， 生成 ml */ 
Var Url = "chklogin.php?key="+key: 
让 ”使 用 xmlhttp 对 象 调用 chkloginphp 页 。*/ 
xmilhttp.open("GET",url,true): 
xmlhttp.onreadystatechange = functionO{ 
if(xmlhttp.readyState 一 4){ 
var msg = Xmlhttp.TesponseText: 
作用 户 没有 登录 */ 
这 msg 一 2){ 
alert(' 请 您 先 登录 ): 
Tetum false; 
}else iftmsg — 3){ 
请 商品 已 添加 沁 
alert(' 该 商品 已 添加 7: 
Tetum false: 
jelsef 
请 “显示 购物 车 */ 
location='index.php?page=shopcar': 
} 
} 
b 
xmlhttp.send(null): 
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(2) 在 chklogin php 文件 中 将 商品 添加 到 购物 车 中 。chklogin php 文件 的 代码 如 下 : 


例 程 32 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\chklogin.php 
<?php 
session start(); 
ee 
* 1 表示 添加 成 功 
* 2 表示 用 户 没有 登录 
* 3 表示 商品 已 添加 过 
* 4 表示 添加 时 出 现 错误 
* 5 表示 没有 商品 添加 
3 
include_once 'system/system.inc.php'; 
Sreback = '0'; 
这 empty($_SESSION[member])){ 
Sreback = "2'; 
}else{ 
Skey = $_GET['key']: 
ifSkey 一 "){ 
Sreback ='5'; 
}else{ 
$id = (inb$_SESSIONTid]; 
S$boo = false: 
S$addshop = arrayO: 
$sql = "select id,shopping from tb_user where id = ".$id; 
Srst = $conn->execute($sq]): 
S$shopcont = S$rst->fields['shopping']: 
这 !empty(SshopconD){ 
Sarr = explode((@'.Sshopconb: 
foreach($arr as $value){ 
if($key 一 $value[O0){ 
Sreback = "3'; 
S$boo = true; 
break: 
V 


} 
这 !$boo){ 


/开启 session 支 持 


// 载 入 数据 库 连接 文件 

// 声 明 返 回 值 

/如 果 session 值 为 空 ， 说 明 没有 登录 
/参见 头 注释 


// 获 取 商 品 id 


// 获 取 登 录 会 员 id 号 

// 设 置 一 个 布尔 变量 ， 初 始 为 false 
// 数 据 库 更 新 数组 

// 生 成 当前 用 户 的 查询 语句 

// 返 回 当前 用 户 记 录 集 

// 找 到 shopping 字 段 

// 如 果 字 段 不 为 空 ， 说 明 已 有 商品 
// 将 字段 用 “@” 拆 分 并 另存 数组 
// 循 环 输出 数组 内 容 

// 判 断 需 要 添加 的 商品 是 否 存 在 
// 参 见 头 注释 

// 同 时 布尔 变量 变 为 true 

/停止 循环 


/如 果 布 尔 值 没有 改变 ， 再 添加 商品 


全 “将 商品 数量 默认 设 为 1， 并 连同 商品 id 保存 到 字 串 中 */ 


$shopcont = '@'.S$key.…1 

请 ”将 商品 字 串 保存 到 更 新 数组 中 */ 
$addshop["shopping'] = $shopcont: 

人 # ”生成 更 新 语句 */ 


SupdateSQL = $conn->GetUpdateSQL(Srst.$addshop); 


人 # ”更 新 数据 表 */ 

这 false 一 $conn->execute($updateSQL)){ 
Sreback = "4'; 

}else{ 
Sreback ="1'; 

由 
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证 ”如果 shopping 字 段 为 空 ， 则 直接 更 新 多 
Stmparr = $key.".1": 
$addshop['shopping'] = $tmparr: 
SupdateSQL = $conn->GetUpdateSQL(Srst,$addshop):; 
if(false 一 $conn->execute($SupdateSQL)){ 
Sreback = "4'; 
}else{ 
Sreback ="1'; 
| 
} 
1 
} 
echo S$reback: 
?> 


通过 分 析 上 述 代码 可 知 ，shopping 字段 保存 的 是 购物 车 中 的 商品 信息 ， 一 条 商品 信息 包括 两 部 分 ， 
即 商品 id 和 商品 数量 ， 其 中 商品 数量 默认 为 1。 两 部 分 之 间 使 用 逗号 “.” 分 隔 ， 如 果 添 加 多 个 商品 ， 
则 每 个 商品 之 间 使 用 “@ ”分隔 。 

成 功 完成 商品 的 添加 操作 后 ， 即 可 进入 到 购物 车 页 面 执行 其 他 的 操作 。 


12.10.3 ”购物 车 展示 


国 ”购物 车 展示 使 用 的 数据 表 : tb commo、tb_user 


购物 车 页 面 分 为 PHP 代码 页 和 Smarty 模板 页 。 在 PHP 代码 页 中 ， 首 先 读 取 tb_user 数据 表 中 
shopping 字段 的 内 容 ， 如 果 字 段 为 空 ， 则 输出 “ 暂 无 商品 ”; 如 果 数 据 库 中 有 数据 ， 则 循环 输出 数据 ， 
并 将 商品 信息 保存 到 数组 中 ， 再 传 给 模板 页 。 购 物 车 页 面 的 代码 如 下 : 


例 程 33 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\myshopcar.php 
<?php 
睛 ” 载 入 数据 库 连 接 文件 和 Smarty 配 置 文件 %/ 
include_once 'system/system.inc.php': 
谨 ”根据 登录 会 员 的 用 户 名 生成 查询 语句 */ 
$sqll = "select id.shopping from tb_user where name =".$_SESSION['member’]."™": 
雍 ”执行 查询 语句 返回 结果 集 */ 
Srst = $conn->execute($sql1): 
雍 ”如 果 字 段 为 空 ， 直 接 输出 “无 商品 信息 ”， 并 结束 运行 了 
这 Srst->fields['shopping] — "){ 
echo "<p>": 
echo "购物 车 中 暂时 没有 商品 !: 
exit(): 


y 

族 ”如 果 有 结果 集 ， 则 将 结果 集 另存 为 数组 i 
Stmparr = $rst->GetAssoc(): 

Scommarr = array(): 


请 ”输出 shopping 字 段 值 3 
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foreach($tmpar as $value){ 

全 ”使 用 “@?” 进 行 分 隔 ， 得 到 数组 */ 

S$tmpnum = explode('@'. $value): 

广 ”得 到 购买 商品 种 类 wh 

$shopnum = count($Stmpnum); 

入 商品 总 价值 */ 

$sum = 0: 

上 ”数组 循环 

foreach($tmpnum as $key => SvD{ 
必 ”使 用 逗号 “,” 拆 分 单一 的 商品 信息 0 
$s_commo = explode(',',$Vv)); 
必 ”根据 保存 的 商品 信息 查找 商品 # 
$sql2 = "select id,name,m price,fold,v_price from tb_commo"; 
$commsql = $sql2." where id = ".$s_commo[0]: 
人 # ”返回 结果 集 */ 
S$commrst = $conn->execute($commsq]): 
证 ”将 结果 集 另存 数组 。 */ 
$arr = $commrst->GetArray(); 
族 ”将 需要 显示 的 信息 处 理 后 保存 到 数组 中 */ 
$ar[0][mum'] = $s_commo[1]:; 
Sarr[Ol['total] = $s_commo[1]*$arr[O]['v_price']: 
$sum += $ar[0]['total']: 
Scommarr[$key] = $arr[0]: 

} 


由 
作 ”将 得 到 的 两 个 数组 和 总 消费 金额 传递 给 模板 */ 
S$smarty->assign('shoparr'. $shopnum): 
S$smarty->assign('‘commarr’.$commarr): 
S$smarty->assign('sum'.$sum): 
Ssmarty->assign("title', 我 的 购物 车 "); 

?> 


商品 的 模板 页 不 仅 负 责 用 户 购 买 商品 信息 的 输出 ， 而 且 还 要 提供 可 以 对 商品 进行 修改 、 删 除 等 操 
作 的 事件 接口 。 模 板 页 的 代码 如 下 : 


例 程 34 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\myshopcar.html 

<!-- 载 入 使 用 到 的 js 脚本 文件 --> 

<script language="javascript" src="js/createxmlhttp.js"></script> 

‘<script language="javascript" src="js/shopcar.js"></script> 

<table> 

<!-- ”购物 车 表单 ”--> 

<form id="myshopcar" name="myshopcar" method="post" action="#"> 
<tr><td> 我 的 购物 车 </td></tt> 
<tr><td >&nbsp:</td><td> 商 品名 称 </td><td> 购 买 数量 </td><td> 市 场 价格 </td> 

<td> 会 员 价格 </td><td> 折 扣 率 </td><td> 合 计 </td></a> 

<!-- ”使 用 foreach 表 单 循环 输出 商品 种 类 -~ 

{foreach key=key item=item from=$commarr} 

<!-- ”删除 复 选 框 ， 名 称 用 数组 表示 ， 每 个 复 选 框 的 值 为 商品 的 id 号 --> 
<tr><td><input id="chk" name="chk[]" type="checkbox" value=" {$itenLid}"></td> 

<!-- 显示 该 商品 的 名 称 -> 


573 


PHP 项 目 开发 案例 全 程 实录 (第 2 版 ) 


<td><div id = "c_ name{f$key}"> &nbsp: {$item.name}</div></td> 
<td> 
<!-- 商品 数量 文本 框 ”每 个 复 选 框 的 名 称 为 “cnum”+key 值 -> 
<input id = " cnum{$key} " name = " cnum{$key} " type = " text " value = ”{Sitemnum} " onkeyup = ”cvp 
( {$key}, {$item.v_price}. {$shoparr})"></td> 
<!-- 商品 的 其 他 信息 --> 
<td><div id="m price{Skey}">é&nbsp: {$item.m price}</div></td> 
"Vv_price{S$key}">&nbsp; {Sitem.v_price}</div></td> 
<td><div id="fold {$key}">&nbsp: {Sitem fold}</div></td> 
<td><div id="total {$key}">é&nbsp. {Sitem.total} </div></td> 
</t> 
<!-- 循环 结束 ”--> 
{/foreach} 
<t> 
<td colspan="3"> 
<!-- 全 选 、 反 选 和 删除 操作 ”--> 
<a href="#" onclick="return alldel(myshopcar)"> 全 选 </a> 
<a href="#" onclick="return overdel(myshopcar):"> 反 选 </a> 
<input type="button" value=" 删 除 选 择 " onClick = return del(myshopcar):></td> 


<td> 
<!-- 继续 购物 --> 

<input type="button" value=" 继 续 购物 " onclick="return conshop(myshopcar)" /></td> 
<td> 


<!-- ”将 当前 用 户 名 保存 到 隐藏 域 中 --> 
<input id="uid" name="uid" type="hidden" value=" {$smarty.session.member}" > 
<!-- 结算， 去 收银 台 --> 
<input type="button" class="btn" value=" 去 收银 台 " onclick="return formset(form)" /></td> 
<td colspan="2"><div id='sum> 共 计 : {$sum}&nbsp; 元 </div></td> 
</t> 
</form> 
</table> 


12.10.4 ”更 改 商 品 数量 


国 更 改 商品 数量 使 用 的 数据 表 : tb commo、tb_user 

对 于 新 添加 的 商品 ， 默 认 的 购买 数量 为 1， 在 购物 车 页 面 可 以 对 商品 的 数量 进行 修改 。 当 商品 数量 
发 生变 化 时 商品 的 “合计 ”金额 和 总 金额 会 自动 发 生 改 变 ， 该 功能 通过 触发 text 文本 域 的 onkeyup 事 
件 调用 cvpO 函 数 来 实现 。cvp0O 函 数 有 3 个 参数 ， 分 别 是 商品 4、 商品 单价 和 商品 类 别 。 

首先 ， 通 过 商品 的 id 可 以 得 到 要 修改 商品 的 相关 表单 和 标签 属性 。 然 后 ， 通 过 商品 单价 和 输入 的 
商品 数量 计算 该 商品 的 合计 金额 。 接 着 ,使 用 for 循环 得 到 其 他 商品 的 合计 金额 。 最后， 将 所 有 的 合计 
金额 累加 ， 并 输出 到 购物 车 页 面 。cvpO 函 数 的 代码 如 下 : 

例 程 35 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\shopcar.js 

上 自动 刷新 总 金额 

* key: 商品 id 

*vpr: 商品 单价 

* shoparr: 商品 种 类 数 
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a 


function cvp(key.vpr.shopar){ 

var n_pre = 'total'; JWdiv 标 签 的 id 前 缀 

var num = 'cnum'tkey.toString(): /根据 key 值 生成 文本 域 的 id 值 

var total =n pretkey.toString(): // 根 据 key 值 生成 div 标 签 的 id 值 

vart number = document.getElementById(num).value: /获取 输入 的 商品 数量 

var tt] = t_number * vpr: // 根 据 商品 数量 和 单价 ， 计 算 商 品 金 额 

document.getElementById(total).innerHTML = tt]: // 更 新 商品 总 金额 

Var sm=0; // 所 有 商品 的 总 金额 ， 初 始 为 0 

for(var i= 0:; i < shoparr: i++){ // 根 据 商品 种 类 数 ， 循 环 得 到 其 他 商品 金额 
Var aaa = document.getElementById(n preti.toString()).innerText: 
sm += parseInt(aaa): // 将 所 有 商品 金额 累加 


} 
document.getElementById('sum').innerHTML = "共计: "+sm+" 元 '; 
} 
这 里 所 更 改 的 商品 数量 ， 并 没有 被 保存 到 数据 库 中 ， 如 果 希 望 保存 ， 那 么 单 击 “ 继 续 购物 ”按钮 
即 可 。 


12.10.5 ”删除 商品 


国 ”删除 商品 使 用 的 数据 表 : tb_commo、tb_user 

当 对 添加 的 商品 不 满意 时 ， 可 以 对 其 进行 删除 。 操 作 流 程 为 : 首先 选中 要 删除 的 商品 前 面 的 复 选 
框 ， 如 果 全 部 删除 ， 则 可 以 单 击 “ 全 选 ”或 “反选 ”按钮 ;然后 单 击 “ 删 除 选择 ”按钮 ， 在 弹出 的 “ 警 
告 ”对 话 框 中 单 击 “ 确 定 ” 按 钮 ， 商 品 将 被 全 部 删除 。 删 除 商品 的 页 面 结果 如 图 12.23 所 示 。 


商品 各 称 购买 数量 市 场 价格 会 员 价格 折扣 军 合计 


商品 名 称 购买 数量 。 | 。 币 场 从 格 会 员 价格 折扣 于 合计 
rc E73 | 83983 5666 T5 666 
E> [|] 共计 : 6666 元 
单 二 确定, 刷新 页 面 


图 12.23 ”删除 商品 流程 
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所 有 的 删除 操作 均 是 通过 js 脚本 文件 shopcarjjs 来 实现 的 ， 相 关 的 函数 包括 alldel0 函 数 、overdel0 
函数 和 del0 函 数 。 


alldel0 和 overdel0 函 数 实现 的 原理 比较 简单 ， 可 通过 触发 onclick 事件 来 改变 复 选 框 的 选中 状态 。 
函数 的 实现 代码 如 下 : 


例 程 36 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\shopcarjs 


入 ”全 部 选择 */ 
function alldel(form){ 
var leng = form.chk.length: // 获 取 复 选 框 数量 
if(leng—undefined){ // 如 果 等 于 undefined， 说 明 只 有 一 个 复 选 框 
if(!form.chk.checked) 
form.chk.checked=true: // 将 复 选 框 置 于 选中 状态 
Jelse{ //undefined， 说 明 有 多 个 复 选 框 
for( vari= 0: i< leng: it+) // 使 用 for 循 环 ， 将 所 有 复 选 框 选中 
if(!form.chk[i].checked) 


form.chk[il].checked = true; 
} 
} 


Teturn false; 


} 
族 ”反选 ， 就 是 将 选中 的 复 选 框 取消 选中 ， 而 未 被 选中 的 复 选 框 则 被 选中 */ 


function overdel(form){ 
var leng = form.chk.length: // 获 取 复 选 框 个 数 
ifleng 一 undefined){ // 如 果 leng 为 undefiend， 说 明 只 有 一 个 复 选 框 
if(!form.chk.checked) 
form.chk.checked=true: 
else 
form.chk.checked=false; 
jelse{ /和 否则， 说明 有 多 个 复 选 框 
for( vari=0:i< leng: it+) // 使 用 for 循 环 ， 对 所 有 复 选 框 操 作 
{ 
if(!form.chk[i].checked) // 根 据 复 选 框 checked 属 性 的 情况 ， 进 行 反 向 选择 
form.chk[i].checked = true: 
else 


form.chk[i].checked = false: 
} 
} 


return false; 


} 


使 用 alldel0 或 overdel0 函 数 选中 复 选 框 后 ， 即 可 调用 del0 函 数 来 实现 删除 功能 。del0 函 数 首先 使 
用 for 循 环 ， 将 被 选中 的 复 选 框 的 value 值 取 出 并 存 成 数组 ， 然 后 根据 数组 生成 url， 并 使 用 xmlhttp 对 
象 调用 该 url， 当 处 理 完毕 后 ， 根 据 返回 值 弹出 提示 或 刷新 本 页 。 该 函数 的 实现 代码 如 下 : 

例 程 37 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\shopcarjs 

启 ”删除 记录 对 

function del(form){ 
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if(!window.confirm(' 是 否 要 删除 数据 ??")){ 


}else{ 
Var leng = form.chk.length: 
这 leng 一 undefined){ 
if(!form.chk.checked){ 
alert(' 请 选取 要 删除 数据 !"): 
}else{ 
rd = form.chk.value; 
var url = "delshop.php?rd="+rd; 
xmlhttp.open("GET".urlLtrne): 


Xmlhttp.onreadystatechange = delnow: 


xmilhttp.send(nulD); 
} 
jelse{ 
var rd=new Array(); 
varj=0; 
for( vari= 0; i< leng; it+) 


if(form.chk[i].checked){ 
rd[j++] = form.chk[il.value: 
} 
这 rd 一 "){ 
alert( 请 选取 要 删除 数据 !; 
}else{ 
Var url = "delshop.php?rd="+rd; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = delnow: 
xmlhttp.send(null): 
} 
} 
b 
Teturn false; 


} 
谨 ”显示 状态 */ 
function delnowO{ 
if(xmlhttp.readyState 一 4){ 
这 xmlhttp.status 一 200){ 
var msg = xmlhttp.responseText; 
iftmsg ="1){ 
alert( 删 除 失败 ): 
}else{ 
alert( 删 除 成 功 ): 
location=(2page=shopcar): 


/返回 复 选 框 的 长 度 


/如 果 只 有 一 个 复 选 框 ， 且 处 于 选中 状态 
// 将 复 选 框 的 value 值 直接 赋 给 变量 rd 

// 根 据 rd 生 成 url 

// 调 用 xmlhttp 对 象 

/调用 delnow0 函 数 


/如果 复 选 框 为 多 个 
/声明 数组 
// 循 环 检查 复 选 框 状 态 


// 将 被 选中 的 复 选 框 的 value 值 存 到 rd 内 


// 解 释 同 上 


// 获 取 xmlhttp 对 象 返回 的 文本 值 
// 如 果 为 1， 说 明 删除 失败 


// 否 则 ， 说 明 删 除 成 功 ， 刷 新 购物 车 
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12.10.6 ”保存 购物 车 


国 ”保存 购物 车 使 用 的 数据 表 : tb_commo、tb_user 

当 用 户 希望 保存 商品 更 改 后 的 商品 数量 时 ， 可 以 单 击 “ 继 续 购物 ”按钮 ， 将 触发 onclick 事件 调用 
conshop0 函 数 保存 数据 ， 该 函数 有 一 个 参数 ， 就 是 当前 表单 的 名 称 。 在 conshop0 函 数 内 ， 根 据 复 选 杠 
和 商品 数量 文本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 

这 里 需要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ,那么 商品 1 的 数量 
就 要 保存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 再 使 用 xmlhttprequest 对 象 调用 url， 最 后 根 
据 回 传 信息 作出 相应 的 判断 。conshop0 函 数 的 代码 如 下 : 

例 程 38 ”代码 位 置 ; 光盘 VTM\12\ShoppingSystemNjsshopcarjs 

族 ”更 改 商品 数量 。 */ 


function conshop(form){ 
varn pre='cnum' // 商 品 数量 文本 域 前 级 
var lang = form.chk.length: // 获 取 复 选 框 的 数量 


必 ”如 果 只 有 一 个 复 选 框 ， 那 么 将 商品 id 和 商品 数量 直接 保存 到 变量 中 */ 
if(lang 一 Undefined){ 
Var fst = form.chk.value; 
Var snd = form.cnum0.value; 


族 ”否则 ， 将 商品 id 和 对 应 的 商品 数量 保存 到 两 个 相应 的 数组 中 */ 


}else{ 
var fst= new Array(): // 商 品 id 数 组 
var snd = new Aray0: // 商 品 数 量 数组 


让 ”循环 获取 复 选 框 的 value 值 和 商品 数量 文本 框 的 value 值 */ 
for(var i= 0:i< lang: iHH{ 
var nm =n preti.toString(): 
Var stmp = document.getElementById(nm).value: 
谨 ”对 商品 数量 文本 框 判 断 ， 不 允许 为 空 ， 而 且 不 允许 为 非 数 字 输 入 */ 
这 stmp — "|isNaN(stmp){ 
alert( 不 允许 为 空 ， 必 须 为 数字 '): 
document.getElementByIdCnm).selectO: 
Tetum false: 


} 
snd[i] = stmp: 
var ftmp = form.chk[i].value: 
fst[i] = fmp: 
} 
} 
Var url = 'changecar.php?fst="+fst+'&snd="+snd: // 生 成 url 
Xmlhttp.open("GET".urltrue): // 调 用 xmlhttp 对 象 
Xmlhttp.onreadystatechange = updatecar: // 调 用 updatecar0 函 数 
xmlhttp.send(null): 
} 
访 ”对 xmlhttp 对 象 的 返回 值 进 行 处 理 。 */ 
function UpdatecarO{ 
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这 xmlhttp readyState — 4){ 
var msg = xmlhttp.responseText: 


iftmsg —'1){ 
location="index.php'; /如 果 操 作成 功 ， 返 回首 页 
}else{ 
alert( 操 作 失 败 ): /否则 ， 提 示 错 误 


} 


} 


在 conshop() 函 数 中 调用 的 changecarphp 页 为 数据 处 理 页 ， 该 页 将 商品 id 和 商品 数量 进行 重新 排 
列 ， 并 保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
例 程 39 ”代码 位 置 ; 光盘 \TM\V12\ShoppingSystemchangecarphp 


<?php 
include_once 'system/system.inc.php'; // 连 接 数 据 库 
$sql = 'select id,shopping from tb_user where id = '(inb$_SESSION['id]: // 生 成 sql 查 询 语句 
Sfst = $_GET['fst]: // 获 取 商 品 id 
$snd=$_ GET['snd']; // 获 取 商 品 数量 
Sreback ="0'; // 设 置 返回 值 ， 默 认为 0 
Srst = $conn->execute($sq]): // 执 行 查询 语句 ， 返 回 查询 结果 集 
Schangecar = array(0): // 创 建 更 新 数组 


if($fst !{=" and $snd (="){ 
族 ”使 用 逗号 “,” 将 商品 id 和 商品 数量 分 隔 ， 并 分 别 保存 到 两 个 临时 数组 中 */ 
Sfarr = explode(',', $fst):; 
$sarr = explode(,.$snd): 
从 临时 数组 */ 
Supcar = array(): 
信使 用 for 循 环 ， 将 商品 id 和 商品 数量 一 一 对 应 ， 重 新 组 合 */ 
for($i = 0: $i < count($farD: SiH){ 
Supcar[$i] = $farr[$i].",".$sarr[$i]: 


} 
局” 如 果 数 组 大 于 1， 说 明 有 多 个 商品 ， 使 用 “@” 将 不 同 商品 分 隔 开 ot 
if(count($farr) > 1){ 
Schangecar['shopping’] = implode(@'.Supcan): 
}else{ 
作 ”如 果 只 有 一 个 商品 ， 则 直接 保存 到 更 新 数组 中 */ 
Schangecar['shopping’] = Supcar[0]: 


族 ”生成 更 新 语句 。 */ 
$UpdateSql = $conn->GetUpdateSQL(Srst, Schangecar,true): 
全 ”更 新 数据 库 */ 
这 false 一 $conn->execute($UpdateSqD)){ 
Sreback = "2'; 
}else{ 
Sreback = "1'; 
} 
9 
echo $reback: 
9 
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12.11 收银 台 模 块 设计 


12.11.1 收银 台 模 块 概述 


当 用 户 停止 浏览 商品 准备 结账 时 , 可 以 单 击 购物 车 页 面 中 的 “去 收银 台 ?按钮 ,该 按钮 将 触发 onclick 
事件 调用 formset0 函 数 显示 订单 页 面 ， 当 用 户 提 交 订单 后 ， 系 统 将 订单 保存 到 数据 表 tb_form 中 ， 同 时 
清空 购物 车 ， 并 显示 订单 信息 提醒 用 户 记录 订单 号 。 当 货款 发 出 后 ， 还 可 以 对 订单 进行 查询 。 收 银 台 
页 面 的 运行 结果 如 图 12.24 所 示 。 


习 nttp-//127.0.0.1 - 收 良 台 - Microsoft Intermet Eglo 


医 了 YU 型 


I 7 


图 12.24 ”收银 台 页 面 运行 结果 


本 节 所 涉及 的 页 面 有 显示 订单 (formset0 函 数 ) 、 填 写 订 单 〈settle php、settle html) 、 提 交 订 单 
Csettle_ chk.php) 、 反 馈 订 单 〈forminfo.php、fominfo .html) 和 查询 订单 5 部 分 。 


12.11.2 ”收银 台 模块 技术 分 析 


在 收银 台 模 块 中 , 通过 ADODB 中 的 方法 完成 订单 信息 的 添加 和 数据 的 更 新 操作 。 在 ADODB 中 ， 
数据 的 添加 和 更 新 可 以 采用 两 种 方法 :第 1 种 方法 直接 定义 INSERT 或 者 UPDATE 语句 ,通过 execute() 
函数 来 执行 添加 或 者 更 新 的 操作 。 第 2 种 方法 使 用 ADODB 提供 的 如 下 函数 来 完成 。 


1. GetlnsertSQL() 函 数 

GetInsertSQLO 函 数 向 数据 库 中 添加 新 的 记录 ， 其 语法 如 下 : 

GetInsertSQL(&Srs. $arrFields) 

参数 $rs 设置 要 添加 记录 的 结果 集 ; 参数 $arrFields 设置 新 记录 的 字段 值 及 字段 名 称 。 

2. GetUpdateSQL() 函 数 

GetUpdateSQL() 函 数 更 新 数据 库 中 指定 的 记录 ， 其 语法 如 下 : 

GetUpdateSQL(&Srs. SarFields. SforceUpdate-false) 

参数 $rs 指定 要 更 新 的 结果 集 ; 参数 $arrFields 指定 更 新 的 字段 和 内 容 ; 参数 $forceUpdate 控制 更 新 
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操作 是 否 执行 ， 如 果 $forceUpdate 为 tue， 无 论 $arrFields 和 Srs 的 值 是 否 相 等 都 将 更 新 。 
12.11.3 ”显示 订单 


国 显示 订单 使 用 的 数据 表 : tb commo、tb_user 

订单 信息 提交 页 面 的 输出 由 formset0 函 数 决 定 , 它 将 商品 信息 进行 整理 , 然后 通过 open0 方 法 打 
开 settle.php 页 来 显示 订单 ， 并 将 整理 后 的 商品 信息 传递 到 settle.php 文件 中 。formsetO 函 数 的 代码 
如 下 : 

例 程 40 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\js\shopcarjs 


function formset(form){ 
var uid = form.uid.value: // 获 取 订 单 提交 人 的 名 称 
varn pre ='cnumy': /商品 数量 文本 前 级 

var lang = form.chk.length; // 复 选 框 个 数 


族 ” 如 果 复 选 框 只 有 一 个 */ 
if(lang — undefined){ 


Var fst = form.chk.value: // 商 品 id 
var snd = form.cnum0.value; // 商 品 数量 
任 ” 当 有 多 个 复 选 框 时 所 
}else{ 
Var fst= new Array0: 


Var snd = new Array0: 
族 ”使 用 fo 循环， 获取 商品 数量 和 商品 id */ 
for(var i= 0; i< lang: iH){ 
var nm =n preti.toString(O:; 
Var stmp = document.getElementById(nm).value: 
if(stnp =—" ||isNaN(stmp){ 
alert( 不 允许 为 空 ， 必 须 为 数字 7): 
document.getElementById(nm).select(); 
Tetum false; 
} 
snd[i] = stmp; 
var ftmp = form.chk[i].value: 
fstfi] = fomp: 
b 


从“ 使 用 open(0 方 法 打开 settle php 页 ， 并 将 当前 用 户 会 员 名 称 、 商 品 id 和 商品 数量 的 值 传 过 去 */ 
open(settle php?uid="+uid+ "gfst="+Hfst+'&snd="+snd,' blank','width=420 height=220',false): 
} 
[DD 说 明 : 因为 open( 方 法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ， 导 致 session 值 传 不 过 去 ， 所 以 这 
里 使 用 隐藏 域 来 传递 用 户 名 称 。 
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12.11.4 填写 订单 


国 ”填写 订单 使 用 的 数据 表 : tb_ commo、tb user、tb form 
settle.php 直接 将 接收 的 值 传 给 settle html 模板 ， 并 载 入 该 模板 。settle php 页 面 的 代码 如 下 : 
例 程 41 代码 位 置 : 光盘 \TM\12\ShoppingSystem\settle.php 


<2php 
include_once 'system/system.inc.php'; // 载 入 Smarty 配 置 文件 
Sfst =$_ GET['fSt]:; /获取 GET 属 性 值 


$snd =$_GET['snd']: 
Suid= $_GET['uid']: 
Ssmarty->assign("title', 结 账 "); // 将 标题 、 数 组 等 数据 传递 给 模板 
S$smarty->assign('fst', $fst): 
S$smarty->assign('snd', $snd): 
S$smarty->assign('uid',$uid); 
$smarty->display('settle.html'): // 显 示 settle.html 模 板 文件 
gy 


settle.html 模板 显示 一 个 表单 ， 该 表单 的 内 容 需要 用 户 来 填写 ， 包 括 收 货 人 、 联 系 电话 等 信息 。 而 
从 PHP 页 传 过 来 的 几 个 变量 则 被 保存 到 隐藏 域 以 传递 到 处 理 页 ， 在 表单 中 将 数据 提交 到 settle_chk.php 
处 理 页 。 


12.11.5 “处理 订单 


国 。 处 理 订单 使 用 的 数据 表 : tb_commo、tb_user、tb_form 

处 理 页 settle_chk.php 获取 表单 中 提交 的 数据 ,根据 用 户 提 交 的 商品 信息 ,重新 查找 数据 表 tb_commo， 
并 从 数据 表 中 提取 商品 信息 保存 到 数组 中 ， 然 后 处 理 页 将 数组 作为 一 条 记录 添加 到 表 tb_form 内 。 

数据 添加 成 功 的 同时 , 处 理 页 会 根据 uid 找到 该 用 户 , 将 shopping 字段 清空 , 最 后 调用 forminfo.php 
页 来 显示 新 添加 的 订单 信息 。settle_chk.php 页 的 代码 如 下 : 


例 程 42 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\settle_chk.php 
<?php 
必 ”获取 表单 值 ， 并 存储 到 添加 数组 中 */ 

include_once 'system/system.inc.php': 

$sql = "select * from tb_form where id = -1"; 

Srst = $conn->execute($sql): 

Saddform = array(): 


S$addform[vendee]=S$ POST[uid]: /| 提交 订单 用 户 
S$addform['commo id] =$_ POST[t]: /购买 商品 这 
S$addform['commo num']=$ POST['snd]: // 购 买 商品 数量 

启 ”根据 商品 id 和 商品 数量 ， 获 取 所 需 数据 。 */ 
$addform[ formid] =timeO /订单 号 
Stmpid = explode('.'.$addform['commo id]): // 将 商品 id 存 为 数组 
S$tmpnm = explode(','.$addform['commo_num"]): /将 商品 数量 存 为 数组 
Snumber = count($tmpid): /获取 商品 数组 长 度 
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广 ” 当 购买 商品 种 类 大 于 1 时 沁 


if($number >1){ 
Stmpna = array0: // 商 品名 称 数组 
Stmpvp = array(): /商品 单价 数组 
Stmpfd = array0O: /商品 折扣 数组 
Stmptt = 0: /商品 总 额 数组 


睛 ”使 用 for 循 环 获取 各 种 商品 的 信息 */ 
for($i= 0; Si<$number: Si++)f 
履 ”生成 查询 语句 ， 执 行 sql 语 句 ， 最 后 返回 查询 结果 集 */ 
Stmpsql = "selectname.V_price.fold from tb_commo where id = ".Stmpid[$i]: 
S$tmprst = $conn->execute($tmpsqD): 
人 “将 每 种 商品 的 信息 放 到 对 应 的 数组 中 。 */ 
Stmpna[$i] = Stmprst->fields[mame']: 
$tmpvp[$i] = $tmprst->fields['v_price']: 
Stmpfd[$i] = $tmprst->fields['fold']: 
S$tmptt += S$tmprst->fields['v_price'] * $tmpnm[$i]; 


} 
店 ”将 获取 的 信息 数组 分 别 生成 长 字符 串 ， 数 组 值 之 间 使 用 逗号 “,” 分 隔 */ 


$addform['commo_name'] = implode(",",$tmpna); /商品 名 称 
$addform['agoprice'] = implode(,Stmpvp): /商品 价格 
$addform[ fold'] = implode(',.Stmpfd): /商品 折扣 
$addform['total] = $tmptt: /商品 总 额 


上 #” 当 购买 一 种 商品 时 */ 

}else if($number — 1){ 
Stmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[0]; 
$tmprst = $conn->execute($tmpsql); 


$addform['commo_name'] = $tmprst->fields['name’]; 
$addform['agoprice'] = $tmprst->fields['v_price']: 
$addform['fold'] = $tmprst->fields['fold]: 
$addform['total'] = Stmprst->fields["v_price'] * $tmpnm[0]: 
儿 没有 商品 时 */ 
}else{ 
echo 'error’; 
exit(); 


} 
证 ”将 表单 的 其 他 信息 存 入 添加 数组 */ 


S$addform['taker] = $_POST['taker’]: /路 货 人 
S$addform['code'] =$_POST['code']: /邮编 

S$addform['tel] =$_ POST['el]: /联系 电话 
$addform['address] = $_POST['address']: /地 址 

Saddform['del method]= $ POST['del]: // 送 货 方式 
S$addform['pay_method'] =$_POST[pay']: // 付 款 方式 
Saddform['state’] ='0' /订单 状态 ，0 为 未 处 理 
SInsertSQL = $conn->GetInsertSQL($rst.$addform): // 生 成 sql 添 加 语句 


这 false 一 $conn->execute($InsertSQL)){ 

启 ”如 果 购 买 失败 ， 返 回 上 层 */ 

echo "<script>alert( 购 买 失败 '):history.back:</script>": 
}else{ 

店 ”如 果 成 功 ， 删 除 购物 车 信息 ， 并 显示 订单 信息  */ 
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?> 


由 于 篇 幅 所 限 ， 有 关 反 馈 订单 和 查询 订单 的 内 容 这 里 不 再 讲解 ， 相 关内 容 请 读者 参考 本 书 光盘 


$updsql = "select * from tb_user where name =".$ POST[uid]."™": 
Supdrst = $conn->execute($updsq]); 


Sarr = array(O: 


$ar['consume'] = $addform['total]: 


$ar['shopping] =": 


S$UpdateSQL = $conn->GetUpdateSQL(S$updrst$arD: 


S$conn->execute($UpdateSQL): 


echo "<script>top.opener.location.reload():</script>"; 
echo "<script>open('forminfo.php?fid=$fid', blank',"width=600 height=450',false):</script>™; 


echo "<script>window.close(): </script>": 


} 


的 源 代码 。 


12.12.1 


后 台 管 理 系统 是 网 站 管理 员 对 商品 、 会 


包括 以 


孔 回回 


12.12 后台 首 页 设计 


后 台 首 页 概述 


下 功能 。 


类 别管 理 模块 : 主要 包括 对 商品 类 别 的 添加 、 修 改 及 删除 操作 。 
商品 管理 模块 : 主要 包括 对 商品 的 添加 、 修 改 、 删 除 及 订单 处 理 操作 。 
用 户 管理 模块 : 主要 包括 对 管理 员 管理 和 会 员 管理 操作 。 其 中 管理 员 管理 是 实现 对 管理 员 的 


添加 、 删 除 和 修改 功能 ， 会 员 管理 则 包括 删除 和 冻结 功能 。 


公告 管理 模块 : 主要 包括 公告 的 添加 及 删除 操作 。 


链接 管理 模块 : 主要 包括 添加 、 修 改 和 删除 友情 链接 操作 。 
台 首 页 的 运行 结果 如 图 12.25 所 示 。 


添加 类 别 IC| | 1209117958 2 银行 转账 查看 请 求 处 理 


员 及 公告 等 信息 进行 统一 管理 的 场所 ， 本 系统 的 后 台 主 要 


四 86 Fm 未 处 理 
。 查看 类 别 加 HIT T3325 | 根 行 科隆 | ”于 闻 来 处理 | 查看 |。 请求 处 覃 
C| | iostess | we pn T3325 | 根 行 和 赎 | 二 部 来 处 理 | 查看 | ”请示 处理 
商品 管理 © sl 3 pn T73326 。 | 要 行 种 账 | 。 平邮 来 外 理 | 查看 请 求 外 于 
0 添加 商品 F| | aas | we 四 21.9 | 银行 基 账 | 。 二 部 来 处理 | 查看 | 。 请求 处 至 
9 查看 商品 三 | | i220no| pa 8866 。 | 银行 种 几 | 进货 上 门 | 未 处 理 | 考 志 | 。 请 示 处 棕 
。 查看 订单 器 | azse| we er 5853 。 | 银行 转账 | 。 平邮 来 处 理 | 查看 | 。 请 地 处 检 
人 上。 8 了 法 慰 


12.25 ”后台 首页 运行 结果 
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12.12.2 后 台 首 页 技术 分 析 


对 后 台 首页 的 设计 和 前 台 首 页 不 同 ， 其 使 用 的 是 框架 布局 。 框 架 布 局 的 特点 是 : 可 以 将 容器 窗口 
划分 为 若干 个 子 窗口 ， 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 ， 网 页 之 间 为 相互 独立 的 ， 没 有 直接 的 关 
联 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 。 框 架 布局 的 好 
处 是 : 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 下 面 来 
具体 看 一 下 框架 布局 的 使 用 格式 及 属性 。 

1， 框 架 布局 格式 

框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 ， 常 用 的 格式 如 下 : 


<html> 
<head> 


</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 


<body> 


</body> 

</noframes> 

</html> 

其 中 <frameset> 和 <frame> 标 签 是 框架 集 标记 ， 而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 
实行 的 一 种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 记 里 的 内 容 ， 让 用 户 能 够 正 
常 浏览 网 页 。 

2. 框架 集 属性 

框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
FRAMESET 标记 的 属性 对 框架 的 结构 进行 设置 。 下 面 给 出 框架 集 的 常用 属性 参数 及 其 说 明和 应 用 举 
例 ， 如 表 12.1 所 示 。 


表 12.1 框架 集 的 常用 属性 


举例 

<frameset cols="2590.100.*" > 

<frame></frame> 

</frameset> 

<frameset rows="25%,100.*" > 
<frame> 
<frame> 

</frameset> 


说 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 取 值 有 
3 种 形式 : 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 


明 


在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 取 值 和 
COLS 类 似 ， 也 是 3 种 形式 


ROWS 
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续 表 
参数 说 了 明 举例 
<framset Cols="25%,*" cols="*" 
ORDE 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边 | frameborder="0"> 
框 ， 默 认 值 ) 或 0 (不 显示 边框 7 
</frameset> 
<framset Cols="25%,*" cols="*" 
指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 默 认 是 无 | framespacing="1"> 
FRAMESPACING | 、 
间隔 的 i 
</frameset> 
<framset cols="25%,*" cols="*" 
指定 边框 的 宽度 ，frameborder 属 性 为 1 时 该 属性 | frameborder="1" border="5"> 
BORDER 
才 有 效 加 
</frameset> 
3. 框架 属性 


使 用 FRAME 标记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 框 架 是 否 包 含 滚动 条 以 及 在 框架 中 显 
示 的 网 页 等 。FRAME 标记 的 常用 属性 参数 及 其 说 明 如 表 12.2 所 示 。 


表 12.2 框架 属性 
参 数 说 有 明 
NAME 指定 框架 的 名 称 
SRC 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、PHP、JSP 等 ) 
FRAMEBODER 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1( 显示 边框 ， 为 默认 ) 或 0( 不 显示 边框 
NORESIZE 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 
SCROLLING 指定 框架 是 否 包 含 滚动 条 。 属 性 可 以 是 yes (有 ) 、no (没有 ) 和 auto (自由 ) 


12.12.3 后台 首 页 实现 过 程 


国 ”后台 首页 使 用 的 数据 表 : tb_ commo、tb_admin、tb_public、tb links、tb_class 


后 台 首 页 的 具体 实现 步骤 如 下 : 
(1) 定义 框架 页 面 main.html， 该 页 面包 含 3 个 文件 : top.html、leftphp 和 default.php。main.html 
页 面 的 代码 如 下 : 


例 程 43 ”代码 位 置 : 光盘 \TM\12\ShoppingSystem\admin\main html 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html: charset=gb2312" /> 

<link rel="stytlesheet" href="css/style.css" /> 

<title> 鹏 逊 电子 商务 后 台 管理 系统 </title> 

</head> 

<!-- ”使 用 frame 框架 布局 -> 

<frameset rows="126.*" cols="*" frameborder="no" border="0" framespacing="0"> 


$586 
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<!--” 头 部 框架 ， 命 名 为 topFrame， 载 入 文件 为 top.html --> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
<frameset rows="*" cols="210,*" framespacine="0" frameborder="no" border="0"> 
<!--” 左 侧 框 架 ， 名 字 为 leftfFrame， 载 入 文件 为 leffphp -> 
<frame src="leftphp" name="leftFrame" frameborder="0" scrolling="auto" noresize="noresize"” id="leftFrame" 
title="leftFrame" /> 
<1-- ”中 部 框架 ， 名 字 为 mainFrame， 载 入 文件 为 default.php --> 
<frame src="default.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
<noframes><body></body></noframes> 
</html> 


(2) left.php 文件 是 一 个 树 形 菜单 ， 应 用 DIV+JavaScript+CSS 来 实现 的 。 首 先 介绍 div 标签 ， 在 
lefthtml 模板 文件 中 ， 其 关键 代码 如 下 : 


例 程 44 ”代码 位 置 ， 光盘 \TM\12\ShoppingSystem\admin\left.html 

<!-- 载 入 css 样 式 和 javascript 脚 本 --> 

<link href="css/left.css" rel="stylesheet" type="text/css" /> 

<script language="javascript" src="js/left.is"></script> 

<!-- “类别 管理 菜单 ， 注 意 加 粗 的 地 方 ”--> 

<div id="type" align="center" onclick="javascript:change(one,type):"> 类 别管 理 </div> 

<!-- 子 菜 单 -> 

<div id="one" style="display: "> 

<div id="addtype" align="center"><a href="addtype.php" target="mainFrame" id="menu"> 添 加 类 别 </a></div> 
<div id="showtype" align="center"><a href="showtype.php" target="mainFrame" id="menu"> 查 看 类 别 </a></div> 
</div> 

<div id="hidediv" align="center"></div> 

<!-- 商品 管理 菜单 ”--> 

<div id="commo" align="center" onclick="javascriptxchange(two,.typej:"> 类 别管 理 </div> 

<div id="two"style="display:none"> 

<!-- 商品 管理 子 菜单 ”“--> 


</div> 


[0D 说 明 : 除了 加 粗 的 过 名 称 和 js 事件 不 同 外 ， 其 他 菜单 的 结构 完全 相同 ， 此 时 只 需 修改 超 链 接 即 可 。 
全 注意 : 除了 第 1 个 类 别 菜单 的 子 菜单 的 display 样式 为 空 外 ， 其 他 几 个 子 菜单 的 div 样式 都 为 display 
三 Done:。 


该 页 面 在 Dreamweaver 中 的 效果 如 图 12.26 所 示 。 
类别 血 尖 


图 12.26 div 树 形 菜单 
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因为 其 他 子 菜单 的 样式 为 display=none， 所 以 只 有 “类 别管 理 ” 子 菜单 是 可 见 的 ， 下 面 为 它 添加 
JavaScript 事件 。leftjs 脚本 文件 的 代码 如 下 : 


例 程 45 ”代码 位 置 光盘 \TM\12\ShoppingSystem\admin\js\leftjs 
谨 函数 功能 : 改变 子 菜单 样式 ， 更 换 当 前 菜单 的 背景 图 片 
* nu: 为 子 菜单 div 标 签 的 id 号 
* ]x: 为 调用 change0 函 数 所 在 的 标签 id 
| 
function change(nu,lx){ 
启 ”如 果 当 前 子 菜单 的 样式 为 none， 则 显示 子 菜 单 。*/ 
if(nu.style.display 一 "none"){ 
nu.style.display = "": 
lx.style.backeround="url(images/main openroot.gif)"; 
/ ”否则 ， 隐 藏 当前 菜单 的 子 菜单 */ 
}else{ 
nu.style.display = "none"; 
Ix.style.background="url(images/main closeroot.gif)"; 


六 


最 后 在 left.css 中 设置 div 的 长 、 宽 等 一 些 默 认 参数 。 一 个 简单 而 又 实用 的 树 形 菜单 就 完成 了 。 
对 于 后 台 的 大 部 分 模块 来 说 ， 其 功能 实现 的 方法 和 开发 步骤 在 前 台 的 模块 设计 中 基本 都 已 经 介绍 
。 由 于 篇 幅 所 限 ， 这 里 不 再 对 后 台 管理 模块 进行 详细 的 讲解 。 


应 


12.13 ”开发 技巧 与 难点 分 析 


全 4 视频 录像 :光盘 \TM\Ix\12\ 开 发 常见 问题 与 解决 .exe 


本 系统 在 开发 和 后 期 测试 的 过 程 中 ， 开 发 人 员 遇 到 了 各 种 各 样 的 疑难 问题 。 这 里 找 出 一 些 比较 常 
见 且 容易 被 忽略 的 问题 加 以 讲解 ， 希 望 能 够 为 初学 者 和 新 手提 供 一 些 帮助 ， 从 而 在 开发 程序 时 少 走 一 些 
弯路 。 


12.13.1 解决 Ajax 的 乱码 问题 


问题 描述 : 当 使 用 Ajax 传递 数据 时 ， 要 么 在 数据 处 理 页 中 数据 不 能 被 正确 处 理 ， 要 么 输出 返回 值 
时 显示 的 是 一 堆 无 法 识别 的 乱码 。 

解决 方法 : 这 是 因为 PHP 在 传递 数据 时 使 用 的 编码 默认 为 UTF-8， 这 就 造成 了 非 英文 字符 不 能 正 
确 传 递 。 解 决 方法 如 下 : 

在 所 有 的 PHP 页 中 都 输入 代码 “Header("Content-Type:text/html;charset=gb2312");”， 这 样 所 有 的 
页 面 即 可 正确 显示 。 
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12.13.2 ”使 用 JS 脚本 获取 、 输 出 标签 内 容 


问题 描述 获取、 更 改 表单 元 素 值 和 特定 标签 内 容 。 

解决 方法 : 使 用 JS 脚本 获取 页 面 内 容 的 方式 主要 有 两 种 ， 第 1 种 是 通过 表单 获取 表单 元 素 的 value 
值 。 格式 为 : 表单 名 称 .元 素 名 .value。 该 方式 只 能 获取 表单 中 的 元 素 值 , 对 于 其 他 标签 元 素 则 无 能 为 力 。 
而 第 2 种 方式 可 以 通过 id 名 来 获取 页 面 中 任意 标签 的 内 容 。 格 式 为 :document.getElementById(id). value; 
或 document.getElementById (id).innerText: 。 

使 用 第 2 种 方式 时 要 注意 , 标签 的 id 名 必须 存在 且 唯一 ， 否 则 就 会 出 现 错误 。 为 标签 内 容 赋值 时 ， 
则 使 用 如 下 的 格式 : 

id.innerHTML = 要 显示 的 内 容 '; 


12.13.3 ”禁用 页 面 缓存 


问题 描述 : 使 用 Ajax 技术 可 以 防止 页 面 刷新 ， 但 有 时 也 会 产生 新 的 问题 。 例 如 ， 在 “会 员 管理 ” 
页 面 ， 如 果 连 续 地 “冻结 ”和 “解冻 ”会 员 ， 那 么 超过 3 次 后 ， 该 功能 将 失效 ， 因 为 在 一 定时 间 内 ， 
如 果 做 相同 的 操作 ， 那 么 xmlhttprequest 对 象 会 执行 缓存 中 的 信息 ， 从 而 造成 操作 失败 。 

解决 办 法 : 使 用 header0 函 数 将 缓存 关闭 。 将 代码 header("CACHE-CONTROL:NO-CACHE"); 添 加 
到 xmlhttprequest 对 象 所 调用 的 处 理 页 的 顶部 即 可 。 


12.13.4 在 新 窗口 中 使 用 session 


问题 描述 ， 使 用 js 的 open() 方 法 打开 新 窗口 时 ， 原 浏览 器 中 的 session 值 不 会 被 传递 到 新 窗口 中 ， 
从 而 造成 数据 查询 失败 。 
解决 方法 : 将 session 值 另存 到 隐藏 域 或 随 着 url 一 起 传递 到 新 窗口 ， 代 码 如 下 : 


<!-- ”在 模板 页 中 ， 将 session 值 赋 给 隐藏 域 --> 

<input id="uid" name="uid" type="hidden" value=" {$smarty.session.id}"> 

必 ”在 js 脚本 中 ， 获 取 到 隐藏 域 value 值 */ 

function getInputO{ 
Var uid = document.getElementById(uid').value: 

启 ”将 获取 的 value 值 通过 url 传 给 新 页 面 */ 
open("operator.php?uid="+uid,’ blank’.".false): 

} 


12.13.5 ”判断 上 传 文件 格式 


问题 描述 : 添加 商品 时 需要 上 传 商品 的 图 片 ， 但 有 时 可 能 会 误 传 非 图 片 格式 的 文件 ， 从 而 导致 上 
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传 图 片 失败 。 
解决 方法 : 这 里 就 自 定义 一 个 函数 来 判断 上 传 文件 的 后 级 ,创建 自 定义 函数 fpostfix0， 函 数 的 代 
码 如 下 : 
话 
* 判 断 文 件 后 级 
*$f type: 人 允许 文件 的 后 组 类 型 数组) 
*$f upfiles: 上 传 文件 名 
人 


function f postfix($f type.$Sf upfiles){ 
Sis_pass = false; 
Stmp_upfiles = split("\.",$f upfiles): // 使 用 split0 函 数 分 隔 文件 
Stmp_num = count($tmp upfiles): // 查 找 文件 后 级 
这 in_array(strtolower(S$tmp upfiles[Stmp_ num - 1]).Sf type)) // 判 断后 缀 是 否 在 允许 列表 内 
Sis_pass = Stmp_upfiles[Stmp num - 1]: /如 果 是 ， 则 将 后 缀 名 赋 给 变量 
Teturn $is_pass; // 返 回 变量 


} 
12.13.6 ”设置 服务 器 的 时 间 


问题 描述 ， 如 果 没 有 对 PHP 的 时 区 进行 设置 ， 那 么 用 户 使 用 日 期 、 时 间 函 数 获取 的 将 是 英国 伦敦 
当地 的 时 间 《〈 即 零 时 区 的 时 间 ) 。 例 如 ， 以 东 八 区 为 例 ， 如 果 当 地 使 用 的 是 北京 时 间 ， 那 么 如 果 没 有 
对 PHP 的 时 区 进行 设置 ， 那 么 用 户 获取 的 时 间 将 比 当 地 的 北京 时 间 少 8 个 小 时 。 

解决 方案 : 要 获取 本 地 当前 的 时 间 必 须 更 改 PHP 语言 中 的 时 区 设置 。 更改 PHP 语言 中 的 时 区 设置 
有 两 种 方法 : 

(1) 在 php.ini 文件 中 ， 去 掉 定位 到 [date] 下 的 “;date.timezone =” 选 项 前 面 的 分 号 ， 并 设置 它 的 

值 为 当地 所 在 时 区 使 用 的 时 间 ， 修 改 内 容 如 图 12.27 所 示 。 

例如 ， 如 果 当 地 所 在 时 区 为 东 作 区， 那么 就 可 以 设置 “date.timezone =” 的 值 为 : PRC、 
Asia/Hong Kong、Asia/Shanghai (上 海 ) 或 者 Asia/yUrumgqi (乌鲁木齐 ) 等 ， 这 些 都 是 东 八 区 的 时 间 。 


; Defines the default timezone used by the 
date Functions | 
; http://php.net/date -timezone 


12.27 设置 PHP 的 时 区 


设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 。 
(2) 在 应 用 程序 中 ， 在 日 期 、 时 间 函 数 之 前 使 用 date_default_timezone_set0 函 数 就 可 以 完成 对 时 
区 的 设置 。date_default_timezone_set() 函 数 的 语法 如 下 : 


date_default timezone set(timezone): 


参数 timezone 为 PHP 可 识别 的 时 区 名 称 ， 如 果 时 区 名 称 PHP 无 法 识别 ， 则 系统 可 以 采用 UTC 


了 
区 | 
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例如 : 设置 北京 时 间 可 以 使 用 的 时 区 包括 PRC (中 华人 民 共 和 国 ) 、Asia/Chongqing (重庆 ) 、 
Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 ) ， 这 几 个 时 区 名 称 是 等 效 的 。 


12.14 Smarty 类 库 技术 专 题 


Smarty 是 一 个 使 用 PHP 编写 的 PHP 模板 引擎 ， 是 目前 业界 最 著名 、 功 能 最 强大 的 一 种 PHP 模板 
引擎 。 它 将 一 个 应 用 程序 分 成 视图 和 逻辑 控制 两 部 分 ， 也 就 是 将 UI (用 户 界 面 ) 和 PHP code (PHP 代 
码 ) 分 离 。 这 样 程序 员 在 修改 程序 时 不 会 影响 到 页 面 设计 ， 而 美工 在 重新 设计 或 修改 页 面 时 也 不 会 影 
响 程序 逻辑 。Smarty 模板 引擎 的 运行 流程 如 图 12.28 所 示 。 


FE 一 次 或 者 模 
已 编译 的 PHP 板 未 改变 则 
和 HTML 混合 页 使 用 原文 件 


图 12.28 Smarty 模板 引擎 的 运行 流程 
12.14.1 Smarty 类 库 下 载 、 安 装 


PHP 没有 内 置 Smarty 模板 类 ， 需 要 单独 下 载 和 配置 。 并 且 Smarty 要 求 服务 器 上 的 PHP 版 本 最 低 
为 4.0.6， 用 户 可 以 去 http://smarty.net/download.php 下 载 最 新 的 Smarty 压缩 包 。 本 书 中 使 用 的 版 本 是 
Smarty-2.6.23， 下 载 地 址 如 图 12.29 所 示 。 
Smarty 类 库 的 安装 步骤 如 下 : 
(1) 将 下 载 的 Smarty 压缩 包 解压 ， 解 压 后 的 文件 夹 如 图 12.30 所 示 。 其 中 有 一 个 libs 目录 ， 该 目 
录 中 包含 Smarty 类 库 的 4 个 核心 文件 smarty.classphp、smarty_Compiler.classphp、config File.class.php 
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和 debug.tpl， 还 包括 intemals 和 plug-ins 两 个 目录 ， 如 图 12.31 所 示 。 


文件 四 ”篇 辑 刀 ”查看 W) 收 章 由 工具 GD) 帮助 如 


@- 日 国 回 的 有 训 克 全 旬 


) 图 Mp /fr erty met doralond ph 


Download 
Note: The Smarty Template Engine requires PHP Version 4.0.6 or later, 
Latest Stable Release 
e Smarty 2.6.25 Ltar.g2) (.zip) May 23rd, 2009 
Previous Releases 


。 Smarty 2,6,24 Ltar.gz) Laln) May 16th, 2009 

e Smarty 2.6.23 (tar.g2) Czip) May 13th, 2009 

。 Smarty 2.6.22 L.tar.g2) .zip) December 17th, 2008 
® Smarty 2.6.21 (tar.gz) (.2ip) December 2nd, 2008 
ea Smarty 2.5.20 (tar.g2) Czip) August 15th, 2008 

。 Smarty 2.6.19 Ctar.gz) Czip) February 11th, 2008 

。 Smarty 2.6.19 Source March 7th, 2007 

®» Smarty 2.6.17 Source March Sth, 2007 

® Smarty 2.6.15 Source Dec 1st, 2006 


图 12.30 ”Smarty-2.6.23 的 压缩 包 图 12.31 Smarty-2.6.23 的 libs 文件 


(2) Smarty 类 库 的 载 入 ， 即 将 libs 文件 夹 复制 到 服务 器 的 根 目录 下 ， 并 为 其 重新 命名 。 一 般 该 目 
录 的 名 称 为 smarty、class 等 ， 这 里 将 libs 文件 夹 重 命名 为 Smarty， 并 且 将 其 复制 到 章 文件 夹 
12\ShoppingSystem\system\smarty 下 ， 如 图 12.32 所 示 。 


TE\12\ShoppingSysten\systens re 


文件 四 查看 WD 收藏 4) 工具 CD 帮助 0 7 
I \AprS erv \rrr \ TRI TP Sr EPE 


文件 天 rp 16:35 


文 音 夹 。。 2010-6-11 16:35 
加 cmtierile dass php PP file 2008-2-11 10:07 
Dabue tl PP file 2008-2-11 10:07 
加 str class. php PP file 2008-2-11 10:07 
加 ser cmeile cess 了 file 2008-2-11 10.07 


区 


12.32 ”Smarty 类 库 的 存储 位 置 
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12.14.2 ” Smarty 类 库 的 配置 


Smarty 类 库 的 配置 步骤 : 

(1) 确定 Smarty 目录 的 位 置 。 因 为 Smarty 类 库 是 通用 的 ， 每 一 个 项 目 都 可 能 会 使 用 到 它 。 所 以 
将 Smarty 存储 在 根 目录 下 , 在 本 项 目 中 将 Smarty 类 库 文件 存储 在 光盘 \12\ShoppingSystem\system\ 文 件 
夹 下 ， 命 名 为 Smarty。 

(2) 新 建 4 个 目录 templates、templates c、configs 和 cache。 其 中 templates 目录 存储 项 目的 模板 
文件 ; templates ec 目录 存储 项 目的 编译 文件 ，configs 目录 存储 项 目的 配置 文件 ，cache 目录 存储 项 目的 
缓存 文件 。 这 4 个 目录 具体 放置 在 什么 位 置 没有 严格 的 规定 ， 只 要 设置 的 路 径 正确 ， 放 在 哪里 都 可 以 。 
在 本 项 目 中 将 创建 的 4 个 文件 夹 存储 在 12\ShoppingSystem\system\ 文 件 夹 下 。 

(3) 创建 配置 文件 。 如 果 要 应 用 Smarty 模板 ， 那 么 就 一 定 要 包含 Smarty 类 库 和 相关 信息 。 配 置 
信息 可 以 存储 在 一 个 单独 的 文件 中 ， 在 需要 时 通过 include 语句 包含 即 可 。 配 置 文件 config.php 的 代码 
如 下 : 


<?php 

必定 义 服务 器 的 绝对 路 径 所 
define(BASE_PATH'.S_SERVER[DOCUMENT ROOT']): 

族 定义 Smarty 模 板 的 绝对 路 径 */ 

define(SMARTY PATH'mrl4\Smarty\: 

必 加 载 Smarty 类 库 文件 。 */ 

require BASE PATH.SMARTY PATH.'Smarty.class.php': 

上 # 实例 化 一 个 Smarty 对 象 */ 

$smarty = new Smarty: 

族 定义 各 个 目录 的 路 径 所 

$smarty->template_dir = '/ /定义 模板 文件 的 存储 位 置 
S$smarty->compile_dir= '/Smarty/templates_c/: /定义 编译 文件 的 存储 位 置 
Ssmarty->config_dir ="./Smarty/configs/': // 定 义 配置 文件 的 存储 位 置 
Ssmarty->cache_dir ="'./Smarty/cache/': // 定 义 缓存 文件 的 存储 位 置 
> 


上 述 配 置 文件 的 参数 说 明 如 下 : 
BASE PATH: 指定 服务 器 的 绝对 路 径 。 
SMARTY PATH: 指定 smarty 模板 的 绝对 路 径 。 
require(): 加 载 Smarty 类 库 文件 Smarty.class.php。 
$smarty: 实例 化 Smart 对 象 。 
$smarty->template dir: 定义 模板 目录 的 存储 位 置 。 
$smarty-> compile_dir: 定义 编译 目录 的 存储 位 置 。 
$smarty-> config dir: 定义 配置 文件 的 存储 位 置 。 
$smarty-> cache_dir: 定义 模板 缓存 目录 。 

以 上 是 最 基本 的 配置 方法 ， 而 在 本 项 目 中 使 用 的 配置 方法 更 加 实用 、 灵 活 ， 在 12.14.3 节 将 对 其 进 
行 详细 讲解 。 


办 办 办 多 办 办 多 轨 
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12.14.3 ”本 项 目 中 Smarty 类 库 的 配置 方法 


本 项 目 中 将 Smarty 配置 方法 封装 到 类 中 ， 并 存储 到 system.smarty.inc.php 文件 中 。 

首先 ， 包 含 Smarty 类 文件 Smarty.classphp。 然 后 ， 定 义 SmartyProject 类 ， 继 承 Smarty 父 类 。 最 
后 ， 定 义 SmartyProject0 方 法 ,设置 Smarty 中 的 模板 文件 〈templates) 、 编 译文 件 〈templates c) 、 配 
置 文件 〈configs) 和 缓存 文件 〈cache) 的 存储 位 置 。system.smarty.inc.php 的 代码 如 下 : 


<2php 

require("../Smarty/Smarty.class.php"): // 调 用 Smarty 文 件 

class SmartyProject extends Smarty{ // 定 义 类 ， 继 承 Smarty 父 类 
function SmartyProjectO{ // 定 义 方法 ， 配 置 Smarty 模 板 


Sthis->template_dir ="./":; // 指 定 模板 文件 存储 在 根 目录 下 
Sthis->compile_dir = "/system/Smarty/templates_c/": // 指 定编 译文 件 存 储 位 置 
Sthis->config dir="./system/Smarty/configs/": 
Sthis->cache_dir = "./system/Smarty/cache/": 

} 


?> 
既然 已 经 将 Smarty 的 配置 方法 存储 到 一 个 类 中 ， 那 么 就 需要 对 类 进行 实例 化 ， 根 据 返 回 的 对 象 名 
称 调用 Smarty 中 的 方法 ， 类 的 实例 化 操作 在 system.inc.php 文件 中 完成 ， 其 返回 对 象 名 称 为 $smarty， 


<?php 
require("system.smarty.inc.php"): // 调 用 类 文件 
S$smarty=new SmartyProject(): // 执 行 类 的 实例 化 操作 
?> 


通过 此 方法 配置 Smarty 模板 的 好 处 是 : 无 论 将 程序 复制 到 哪个 服务 器 下 执行 ， 都 不 需要 更 改 服务 
器 或 者 Smarty 文件 的 绝对 路 径 ， 程 序 就 可 以 直接 运行 。 


12.15 本 章 总 结 


本 章 使 用 Smarty、ADODB、Ajax 等 目前 的 主流 技术 ， 实 现 一 个 电子 商务 平台 从 系统 分 析 到 最 后 
发 布 的 全 过 程 。 希 望 读者 能 通过 这 个 项 目 实例 ， 把 前 面 所 学 到 的 各 种 技术 消化 吸收 、 融 会 贯通 ， 并 能 
够 学 以 臻 用， 举一反三 。 


